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.plugin.table; 017 018import java.util.Calendar; // 7.0.1.3 (2018/11/12) 019 020import org.opengion.hayabusa.db.AbstractTableFilter; 021import org.opengion.hayabusa.db.DBTableModel; 022 023import org.opengion.fukurou.util.ErrorMessage; 024import org.opengion.fukurou.util.StringUtil; 025import org.opengion.fukurou.util.HybsDateUtil; // 7.0.1.3 (2018/11/12) 026import org.opengion.fukurou.system.DateSet; // 7.0.1.3 (2018/11/12) 027 028/** 029 * TableFilter_KEY_BREAK は、TableFilter インターフェースを継承した、DBTableModel 処理用の 030 * 実装クラスです。 031 * 032 * ここでは、指定のカラムに対して、キーブレイクが発生したときのデータ処理方法を指定できます。 033 * 主として、グルーピング処理を行うのですが、ソートされデータの並び順で、キーブレイクするため、 034 * 同一キーが存在していても、並び順が離れている場合、別のキーとしてブレイクします。 035 * 036 * GROUP_KEY : キーブレイクの判定を行うカラムを、CSV形式で設定します。 037 * OUT_TYPE : 出力するデータのタイプを指定します。 038 * first : 最初のデータ(ブレイク直後のデータ)を出力します。(初期値) 039 * last : 最後のデータ(ブレイク直前のデータ)を出力します。 040 * range : 最初のデータと最後のデータを出力します。 041 * 042 * firstは、キーブレイク時のデータを残します。つまり、キーの最初に現れたデータです。 043 * lastは、キーブレイクの直前のデータを残します。これは、同一キーの最後のデータということになります。 044 * rangeは、firstと、last つまり、同値キーの最初と最後のデータを残します。 045 * 046 * もし、キーが、1行だけの場合、firstも、lastも、同じ行を指すことになります。 047 * その場合、rangeは、その1行だけになります(2行出力されません)。 048 * 049 * 例:機種と日付と、状況Fがあったとして、日付、機種、状況F でソートし、機種をグループキー、 050 * 状況Fをブレイクキーとすれば、日付の順に、機種の中で、状況Fがブレークしたときのみ、 051 * データを残す、ということが可能になります。7.0.0.1 (2018/10/09) Delete 052 * 053 * OUT_TYPE に、lastか、range を指定した場合のみ、最大、最小、平均、中間、個数の集計処理が行えます。 054 * これらの設定は、指定のカラムのデータに反映されます。 055 * MIN_CLM : キーブレイク時に、指定のカラムの最小値をデータに書き込みます。 056 * MAX_CLM : キーブレイク時に、指定のカラムの最大値をデータに書き込みます。 057 * AVG_CLM : キーブレイク時に、指定のカラムの平均値をデータに書き込みます。 058 * MID_CLM : キーブレイク時に、指定のカラムの最小値と最大値の中間の値をデータに書き込みます。 059 * DIF_CLM : キーブレイク時に、指定のカラムの最大値から最小値を引いた値(差)をデータに書き込みます。8.0.1.2 (2021/11/19) 060 * CNT_CLM : キーブレイク時に、指定のカラムのデータ件数をデータに書き込みます。 061 * 062 * これらのカラムの値は、数値で表現できるもので無ければなりません。 063 * 例えば、20180101000000 のような、日付でも数字のみなら、OKです。 064 * 065 * 8.0.1.2 (2021/11/19) DIF_CLM 差分計算 066 * 8桁か14桁で、先頭"20"の場合は、日付型と判定します。 067 * その場合、8桁は、経過日数を返し、14桁は、MM/dd HH:mm 形式で返します。 068 * 069 * パラメータは、tableFilterタグの keys, vals にそれぞれ記述するか、BODY 部にCSS形式で記述します。 070 * 071 * @og.formSample 072 * ●形式: 073 * ① <og:tableFilter classId="KEY_BREAK" 074 * keys="GROUP_KEY,OUT_TYPE" 075 * vals='"CLM5,CLM6....",first' /> 076 * 077 * ② <og:tableFilter classId="KEY_BREAK" > 078 * { 079 * GROUP_KEY : CLM5,CLM6.... ; 080 * OUT_TYPE : first ; 081 * } 082 * </og:tableFilter> 083 * 084 * @og.rev 6.7.9.1 (2017/05/19) 新規追加 085 * @og.rev 7.0.0.1 (2018/10/09) グループで、まとめる処理を止めます。 086 * @og.rev 7.0.1.1 (2018/10/22) ロジック見直し 087 * 088 * @version 6.7 2017/05/19 089 * @author Kazuhiko Hasegawa 090 * @since JDK1.8, 091 */ 092public class TableFilter_KEY_BREAK extends AbstractTableFilter { 093 /** このプログラムのVERSION文字列を設定します。 {@value} */ 094 private static final String VERSION = "8.0.1.2 (2021/11/19)" ; 095 096 /** 8.0.1.2 (2021/11/19) 日単位変換係数 */ 097 private static final int MILLIS_OF_DAY = 1000 * 60 * 60 * 24; 098 099 /** 100 * デフォルトコンストラクター 101 * 102 * @og.rev 6.7.9.1 (2017/05/19) 新規追加 103 * @og.rev 7.0.0.1 (2018/10/09) 最小,最大,平均,件数 を集計するためのキーワード追加 104 * @og.rev 7.0.1.3 (2018/11/12) MID_CLM(最小値と最大値の中間の値)のキーワード追加 105 * @og.rev 8.0.1.2 (2021/11/19) DIF_CLM(最大値から最小値を引いた値(差))のキーワード追加 106 */ 107 public TableFilter_KEY_BREAK() { 108 super(); 109 initSet( "GROUP_KEY" , "キーブレイクの判定を行うカラムを、CSV形式で設定します。" ); 110 initSet( "OUT_TYPE" , "出力するデータのタイプを指定[first/last/range]を指定します。(初期値:first 最初のデータ)" ); 111 initSet( "MIN_CLM" , "キーブレイク時に、指定のカラムの最小値をデータに書き込みます。" ); 112 initSet( "MAX_CLM" , "キーブレイク時に、指定のカラムの最大値をデータに書き込みます。" ); 113 initSet( "AVG_CLM" , "キーブレイク時に、指定のカラムの平均値をデータに書き込みます。" ); 114 initSet( "MID_CLM" , "キーブレイク時に、指定のカラムの最小値と最大値の中間の値をデータに書き込みます。" ); 115 initSet( "DIF_CLM" , "キーブレイク時に、指定のカラムの最大値から最小値を引いた値(差)をデータに書き込みます。" ); // 8.0.1.2 (2021/11/19) 116 initSet( "CNT_CLM" , "キーブレイク時に、指定のカラムのデータ件数をデータに書き込みます。" ); 117 } 118 119 /** 120 * DBTableModel処理を実行します。 121 * 122 * @og.rev 6.7.9.1 (2017/05/19) 新規追加 123 * @og.rev 7.0.0.1 (2018/10/09) 最小,最大,平均,件数 を集計するためのキーワード追加 124 * @og.rev 7.0.1.1 (2018/10/22) ロジック見直し 125 * @og.rev 7.0.1.3 (2018/11/12) MID_CLM(最小値と最大値の中間の値)のキーワード追加 126 * @og.rev 7.2.4.0 (2020/05/11) MIN_CLMとMAX_CLMが不定の場合は、ゼロ文字列をセットします。 127 * @og.rev 8.0.1.2 (2021/11/19) DIF_CLM(最大値から最小値を引いた値(差))のキーワード追加 128 * 129 * @return 処理結果のDBTableModel 130 */ 131 public DBTableModel execute() { 132 final DBTableModel table = getDBTableModel(); 133 final DBTableModel rtnTbl = table.newModel(); // 削除ではなく、追加していきます。 134 final int rowCnt = table.getRowCount(); 135 if( rowCnt == 0 ) { return rtnTbl; } // 7.0.1.3 (2018/11/12) row<=rowCnt を追加したので、0件なら即終了 136 137 final String[] brkClms = StringUtil.csv2Array( getValue( "GROUP_KEY" ) ); 138 139 // 7.0.0.1 (2018/10/09) 最小,最大,平均,件数 を集計するためのキーワード追加 140 final String outType = StringUtil.nval( getValue( "OUT_TYPE" ), "first" ) ; 141 142 final boolean useFirst = "first".equalsIgnoreCase( outType ) || "range".equalsIgnoreCase( outType ); // firstかrange時に使用 143 final boolean useLast = "last".equalsIgnoreCase( outType ) || "range".equalsIgnoreCase( outType ) ; // lastかrange 時に使用 144 145 // 7.0.0.1 (2018/10/09) 最小,最大,平均,件数 を集計するためのキーワード追加(useLast=true のときのみ使用) 146 final int minClmNo = table.getColumnNo( getValue( "MIN_CLM" ), false ) ; // カラムが存在しなければ、-1 147 final int maxClmNo = table.getColumnNo( getValue( "MAX_CLM" ), false ) ; // カラムが存在しなければ、-1 148 final int avgClmNo = table.getColumnNo( getValue( "AVG_CLM" ), false ) ; // カラムが存在しなければ、-1 149 final int midClmNo = table.getColumnNo( getValue( "MID_CLM" ), false ) ; // 7.0.1.3 (2018/11/12) カラムが存在しなければ、-1 150 final int difClmNo = table.getColumnNo( getValue( "DIF_CLM" ), false ) ; // 8.0.1.2 (2021/11/19) カラムが存在しなければ、-1 151 final int cntClmNo = table.getColumnNo( getValue( "CNT_CLM" ), false ) ; // カラムが存在しなければ、-1 152 153 final int[] brkClmNo = new int[brkClms.length]; // ブレイクキーカラムの番号 154 155 for( int i=0; i<brkClms.length; i++ ) { 156 brkClmNo[i] = table.getColumnNo( brkClms[i],false ); // カラムが存在しなければ、-1 157 } 158 159 // 7.0.0.1 (2018/10/09) 最小,最大,平均,件数 を集計するためのキーワード追加(useLast=true のときのみ使用) 160 double minData = Double.POSITIVE_INFINITY ; // 仮数部の桁数の限界は15桁なので、日付型(14桁)は、処理できる。 161 double maxData = Double.NEGATIVE_INFINITY ; 162 double total = 0.0 ; 163 int cntData = 0 ; // 164 boolean isLong = true; // データに、少数点以下をつけるかどうかの判定です。 165 double midMin = Double.POSITIVE_INFINITY ; 166 double midMax = Double.NEGATIVE_INFINITY ; 167 double difMin = Double.POSITIVE_INFINITY ; // 8.0.1.2 (2021/11/19) 168 double difMax = Double.NEGATIVE_INFINITY ; // 8.0.1.2 (2021/11/19) 169 170 String oldBlkKeys = null; // 前回ブレイクキーの値 171 172 String[] oldData = null; 173 // 7.0.1.3 (2018/11/12) 最後のデータの処理を行うために、row<=rowCnt と1回余計に回します。 174 for( int row=0; row<=rowCnt; row++ ) { 175 final String[] data = row == rowCnt ? null : table.getValues( row ); // row<=rowCnt の影響 176 try { 177 final String brkKeys = getKeys( brkClmNo , data ); // ブレークキー(data==nullの場合、ゼロ文字列) 178 if( !brkKeys.equalsIgnoreCase( oldBlkKeys ) ) { // キーブレイク 179 if( row>0 ) { 180 // 7.2.4.0 (2020/05/11) MIN_CLMとMAX_CLMが不定の場合は、ゼロ文字列をセットします。 181 if( minClmNo >= 0 ) { 182 if( minData == Double.POSITIVE_INFINITY ) { 183 oldData[minClmNo] = ""; // 7.2.4.0 (2020/05/11) 184 } 185 else { 186 oldData[minClmNo] = isLong ? String.valueOf( Math.round( minData ) ) : String.valueOf( minData ) ; 187 } 188 } 189 if( maxClmNo >= 0 ) { 190 if( maxData == Double.NEGATIVE_INFINITY ) { 191 oldData[maxClmNo] = ""; // 7.2.4.0 (2020/05/11) 192 } 193 else { 194 oldData[maxClmNo] = isLong ? String.valueOf( Math.round( maxData ) ) : String.valueOf( maxData ) ; 195 } 196 } 197 if( avgClmNo >= 0 ) { oldData[avgClmNo] = String.format( "%.3f", total/cntData ); } 198 if( midClmNo >= 0 ) { oldData[midClmNo] = getMiddle( midMin,midMax ); } 199 if( difClmNo >= 0 ) { oldData[difClmNo] = getDifference( difMin,difMax ); } // 8.0.1.2 (2021/11/19) 200 if( cntClmNo >= 0 ) { oldData[cntClmNo] = String.valueOf( cntData ); } 201 202 minData = Double.POSITIVE_INFINITY ; 203 maxData = Double.NEGATIVE_INFINITY ; 204 total = 0.0 ; 205 midMin = Double.POSITIVE_INFINITY ; 206 midMax = Double.NEGATIVE_INFINITY ; 207 difMin = Double.POSITIVE_INFINITY ; // 8.0.1.2 (2021/11/19) 208 difMax = Double.NEGATIVE_INFINITY ; // 8.0.1.2 (2021/11/19) 209 210 if( useLast ) { 211 // useFirst=true で、cntData == 1 の場合は、First行は削除します(1件を2件に増やさない)。 212 if( useFirst ) { 213 final int rCnt = rtnTbl.getRowCount(); 214 if( cntData == 1 ) { // 1行しかない場合は、First行は削除します(1件を2件に増やさない) 215 rtnTbl.removeValue( rCnt-1 ); 216 } 217 else { 218 final String[] fstData = rtnTbl.getValues( rCnt-1 ); // 前のデータ=First行に、最大、最小等のデータを反映させます。 219 if( minClmNo >= 0 ) { fstData[minClmNo] = oldData[minClmNo]; } 220 if( maxClmNo >= 0 ) { fstData[maxClmNo] = oldData[maxClmNo]; } 221 if( avgClmNo >= 0 ) { fstData[avgClmNo] = oldData[avgClmNo]; } 222 if( midClmNo >= 0 ) { fstData[midClmNo] = oldData[midClmNo]; } 223 if( difClmNo >= 0 ) { fstData[difClmNo] = oldData[difClmNo]; } // 8.0.1.2 (2021/11/19) 224 if( cntClmNo >= 0 ) { fstData[cntClmNo] = oldData[cntClmNo]; } 225 } 226 } 227 228 rtnTbl.addColumnValues( oldData ); // ブレイクした一つ前=最後のデータ 229 } 230 if( row == rowCnt ) { break; } // 最後のデータの処理を行うために、row<=rowCnt と1回余計に回します。 231 } 232 233 if( useFirst ) { // useLast=true で、cntData == 1 の場合は、登録しません 234 rtnTbl.addColumnValues( data ); // ブレイク時のデータを登録します。 235 } 236 237 oldBlkKeys = brkKeys; 238 cntData = 0 ; 239 } 240 oldData = data; // 一つ前のデータ 241 cntData++; // 毎回、カラムのある無しを判定するより、早そうなので常にカウントしておきます。 242 243 // ブレイク時も集計処理は行います。 244 if( minClmNo >= 0 && !StringUtil.isNull( data[minClmNo] ) ) { 245 if( isLong && data[minClmNo].indexOf( '.' ) >= 0 ) { isLong = false; } // 一度、false になると、戻らない。 246 minData = Math.min( minData, Double.parseDouble( data[minClmNo] ) ); 247 } 248 if( maxClmNo >= 0 && !StringUtil.isNull( data[maxClmNo] ) ) { 249 if( isLong && data[maxClmNo].indexOf( '.' ) >= 0 ) { isLong = false; } // 一度、false になると、戻らない。 250 maxData = Math.max( maxData, Double.parseDouble( data[maxClmNo] ) ); 251 } 252 if( avgClmNo >= 0 && !StringUtil.isNull( data[avgClmNo] ) ) { 253 total += Double.parseDouble( data[avgClmNo] ); 254 } 255 if( midClmNo >= 0 && !StringUtil.isNull( data[midClmNo] ) ) { 256 final double mid = Double.parseDouble( data[midClmNo] ); 257 midMin = Math.min( midMin, mid ); 258 midMax = Math.max( midMax, mid ); 259 } 260 if( difClmNo >= 0 && !StringUtil.isNull( data[difClmNo] ) ) { // 8.0.1.2 (2021/11/19) 261 final double dif = Double.parseDouble( data[difClmNo] ); 262 difMin = Math.min( difMin, dif ); 263 difMax = Math.max( difMax, dif ); 264 } 265 } 266 catch( final RuntimeException ex ) { // そのまま、継続して処理を行う。 267 // 6.5.0.1 (2016/10/21) ErrorMessage をまとめるのと、直接 Throwable を渡します。 268 makeErrorMessage( "TableFilter_KEY_BREAK Error",ErrorMessage.NG ) 269 .addMessage( row+1,ErrorMessage.NG,"KEY_BREAK" , StringUtil.array2csv( data ) ) 270 .addMessage( ex ); 271 } 272 } 273 274 return rtnTbl; 275 } 276 277 /** 278 * 最小値と最大値の中間の値の文字列を作成します。 279 * 280 * 特殊系で、8桁か、14桁の場合、日付文字として中間の日付を求めます。 281 * 282 * @og.rev 7.0.1.3 (2018/11/12) MID_CLM(最小値と最大値の中間の値)のキーワード追加 283 * 284 * @param min 最小値 285 * @param max 最大値 286 * @return 中間の値の文字列 287 */ 288 private String getMiddle( final double min , final double max ) { 289 final String minStr = String.valueOf( Math.round( min ) ); // 14桁の場合、2.0181103000000E13 見たいな表記になるため。 290 final String maxStr = String.valueOf( Math.round( max ) ); 291 final int minLen = minStr.length(); 292 293 final String midStr ; 294 // 2000 年問題!? 先頭が "20" の場合は、日付型と判定する。 295 if( minLen == maxStr.length() && ( minLen == 8 || minLen == 14 ) 296 && minStr.startsWith("20") && maxStr.startsWith("20") ) { 297 final Calendar minCal = HybsDateUtil.getCalendar( minStr ); 298 final Calendar maxCal = HybsDateUtil.getCalendar( maxStr ); 299 final long midTim = ( maxCal.getTimeInMillis() + minCal.getTimeInMillis() ) / 2 ; 300 301 if( minLen == 8 ) { 302 midStr = DateSet.getDate( midTim , "yyyyMMdd" ); 303 } 304 else { // 14桁しかありえない 305 midStr = DateSet.getDate( midTim , "yyyyMMddHHmmss" ); 306 } 307 } 308 else { 309 midStr = String.format( "%.3f", ( max + min ) / 2.0 ); // 日付型でなければ、minStr,maxStr は使わないので。 310 } 311 312 return midStr; 313 } 314 315 /** 316 * 最大値から最小値を引いた値(差)の文字列を作成します。 317 * 318 * 特殊系で、8桁か、14桁の場合、日付文字として経過日数を求めます。 319 * 320 * @og.rev 8.0.1.2 (2021/11/19) DIF_CLM(最大値から最小値を引いた値(差))のキーワード追加 321 * 322 * @param min 最小値 323 * @param max 最大値 324 * @return 最大値から最小値を引いた値(差)の文字列 325 */ 326 private String getDifference( final double min , final double max ) { 327 final String minStr = String.valueOf( Math.round( min ) ); // 14桁の場合、2.0181103000000E13 見たいな表記になるため。 328 final String maxStr = String.valueOf( Math.round( max ) ); 329 final int minLen = minStr.length(); 330 331 final String midStr ; 332 // 2000 年問題!? 先頭が "20" の場合は、日付型と判定する。 333 if( minLen == maxStr.length() && ( minLen == 8 || minLen == 14 ) 334 && minStr.startsWith("20") && maxStr.startsWith("20") ) { 335 final Calendar minCal = HybsDateUtil.getCalendar( minStr ); 336 final Calendar maxCal = HybsDateUtil.getCalendar( maxStr ); 337 338 final long difTim = maxCal.getTimeInMillis() - minCal.getTimeInMillis() ; 339 340 if( minLen == 8 ) { 341 midStr = String.format( "%d", (int)difTim/MILLIS_OF_DAY ); 342 } 343 else { // 14桁しかありえない 344 midStr = DateSet.getDate( difTim , "MM/dd HH:mm" ); 345 } 346 } 347 else { 348 midStr = String.format( "%.3f", max - min ); // 日付型でなければ、minStr,maxStr は使わないので。 349 } 350 351 return midStr; 352 } 353 354 /** 355 * キーの配列アドレスと、1行分のデータ配列から、キーとなる文字列を作成します。 356 * 357 * @og.rev 6.7.9.1 (2017/05/19) 新規追加 358 * @og.rev 7.0.1.3 (2018/11/12) 最後のデータの処理を行うために、row<=rowCnt と1回余計に回す対応 359 * 360 * @param clms キーの配列アドレス 361 * @param rowData 1行分のデータ配列 362 * @return キーとなる文字列 363 */ 364 private String getKeys( final int[] clms , final String[] rowData ) { 365 if( rowData == null ) { return ""; } // rowData がnull の場合は、キーブレイクとなる 366 367 final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE ); 368 // 7.2.9.4 (2020/11/20) PMD:This for loop can be replaced by a foreach loop 369 for( final int clm : clms ) { 370 if( clm >= 0 ) { 371 buf.append( rowData[clm] ).append( ':' ); 372 } 373 } 374 375 return buf.toString(); 376 } 377}