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.fukurou.util;
017
018import java.util.List;                                                                                          // 8.5.4.2 (2024/01/12) PMD 7.0.0 LooseCoupling
019import java.util.ArrayList;
020import java.util.Map;
021import java.util.HashMap;
022
023import static org.opengion.fukurou.system.HybsConst.CR;                         // 6.1.0.0 (2014/12/26) refactoring
024import static org.opengion.fukurou.system.HybsConst.BUFFER_MIDDLE;      // 6.1.0.0 (2014/12/26) refactoring
025import org.opengion.fukurou.system.OgRuntimeException ;         // 6.4.2.0 (2016/01/29)
026
027/**
028 * SystemParameter は、{@XXXX} 文字列を処理するクラスです。
029 * このクラスでは、{@XXXX} 文字列を別の文字列と置き換えることや、
030 * 予め予約されている予約語 {@DATE.XXXX} 文字列を置き換えます。
031 * 通常の {@XXXX} 文字列の置き換えは、キーと値のペアを、HybsEntry オブジェクトに
032 * セットして、その配列を受け取って処理します。
033 *
034 * 以下の値はあらかじめ、動的に作成されます。
035 * ・DATE.YMD       8byte の今日のシステム日付(yyyyMMdd)
036 * ・DATE.YMDH    14byte の今日のシステム日時(yyyyMMddHHmmss)
037 * ・DATE.HMS       6byte の今日のシステム時間(HHmmss)
038 *
039 * @og.group ユーティリティ
040 *
041 * @version  4.0
042 * @author   Kazuhiko Hasegawa
043 * @since    JDK5.0,
044 */
045public final class SystemParameter {
046
047        private final String    original ;
048
049        private final String[] clms;
050        private final String[] formats;
051
052        /**
053         *  {@XXXX} の特殊文字を含む文字列を、置き換えます。
054         * 対象外の文字列は、そのまま、残されます。
055         *
056         * @og.rev 5.1.8.0 (2010/07/01) パース方法見直し(StringTokenizerでは、{@XXXX}が連続してある場合に対応できない)
057         * @og.rev 5.3.2.0 (2011/02/01) original データを、パース結果を利用するように修正する。
058         * @og.rev 5.3.4.0 (2011/04/01) {@DATE.XXXX} を処理できるように機能追加
059         * @og.rev 5.3.5.0 (2011/05/01) {@SYS.XXXX} は、廃止
060         * @og.rev 5.5.7.2 (2012/10/09) rightNow をCalendarオブジェクト ではなく、String時刻とします。
061         * @og.rev 5.5.7.2 (2012/10/09) HybsDateUtil を利用するように修正します。
062         *
063         * @param       orig    変換する文字列(オリジナル)
064         */
065        public SystemParameter( final String orig ) {
066                if( orig == null || orig.isEmpty() || orig.indexOf( "{@" ) < 0 ) {
067                        clms     = null;
068                        formats  = null;
069                        original = orig;                // 5.3.2.0 (2011/02/01)
070                }
071                else {
072                        final StringBuilder buf = new StringBuilder(orig.length());             // 5.3.2.0 (2011/02/01)
073
074//                      final ArrayList<String> fmtList = new ArrayList<>();
075//                      final ArrayList<String> clmList = new ArrayList<>();
076                        final List<String> fmtList = new ArrayList<>();         // 8.5.4.2 (2024/01/12) PMD 7.0.0 LooseCoupling
077                        final List<String> clmList = new ArrayList<>();         // 8.5.4.2 (2024/01/12) PMD 7.0.0 LooseCoupling
078
079                        // 5.1.8.0 (2010/07/01) パース方法見直し
080                        int start = 0;
081                        int index = orig.indexOf( "{@" );
082                        String val ;
083                        while( index >= 0 ) {
084                                val = orig.substring( start, index );                                   // 5.3.4.0 (2011/04/01)
085                                buf.append(  val );
086                                fmtList.add( val );
087                                final int end = orig.indexOf( '}',index );
088                                if( end < 0 ) {
089                                        final String errMsg = "{@ と } との対応関係がずれています。" + CR
090                                                                + "str=[" + orig + "] : index=" + index ;
091                                        throw new OgRuntimeException( errMsg );
092                                }
093                                final String param = orig.substring( index+2,end );
094                                if( param.startsWith( "DATE." ) ) {             // 5.3.5.0 (2011/05/01) {&#064;SYS.XXXX} は、廃止
095                                        val = getDateFormat( param.substring( 5 ) );    // 5.5.7.2 (2012/10/09) HybsDateUtil を利用時に "DATE." は不要
096                                        clmList.add( null );            // パースした場合は、clmList は、使用しない。
097                                        buf.append( val );
098                                }
099                                else {
100                                        clmList.add( param );
101                                        buf.append( "{@" ).append( param ).append( '}' );               // 元のままの文字列を生成          // 6.0.2.5 (2014/10/31) char を append する。
102                                }
103                                start = end+1;
104                                index = orig.indexOf( "{@",start );
105                        }
106                        val = orig.substring( start, orig.length() );                           // 5.3.4.0 (2011/04/01)
107                        buf.append(  val );
108                        fmtList.add( val );
109
110                        original = buf.toString();              // 5.3.2.0 (2011/02/01)
111                        if( original.indexOf( "{@" ) < 0 ) {
112                                clms     = null;
113                                formats  = null;
114                        }
115                        else {
116//                              clms    = clmList.toArray( new String[clmList.size()] );
117//                              formats = fmtList.toArray( new String[fmtList.size()] );
118                                clms    = clmList.toArray( new String[0] );     // 8.5.4.2 (2024/01/12) PMD 7.0.0 OptimizableToArrayCall 対応
119                                formats = fmtList.toArray( new String[0] );     // 8.5.4.2 (2024/01/12) PMD 7.0.0 OptimizableToArrayCall 対応
120                        }
121                }
122        }
123
124        /**
125         * 日付関係の情報を簡易的に取り出す処理を行います。
126         *
127         * 引数は、"KEY AA(第1引数) BB(第2引数) CC(第3引数)" という状態で受け取ります。
128         * 引数は、なくてもかまいません。ただし、引数にリクエストパラメータ(@で始まる引数)は使えません。
129         *
130         * 処理の詳細は、{@link org.opengion.fukurou.util.HybsDateUtil#getDateFormat( String,String,String,int ) }
131         * または、{@link org.opengion.hayabusa.taglib.CommonTagSupport#getDateFormat( String ) } を
132         * 参照してください。
133         *
134         * @og.rev 5.3.4.0 (2011/04/01) 新規追加
135         * @og.rev 5.5.7.2 (2012/10/09) HybsDateUtil を利用するように修正します。
136         * @og.rev 5.5.8.2 (2012/11/09) prmA の判定に、null と ゼロ文字列を判定する。
137         * @og.rev 5.7.4.1 (2014/03/14) AA 引数の@解析後のコマンド判定方法を、8ケタ以下から先頭が数字以外に変更します。
138         * @og.rev 5.7.4.1 (2014/03/14) taglib.CommonTagSupport#getDateFormat( String ) と処理を合わせます。
139         * @og.rev 5.7.4.2 (2014/03/20) リクエストパラメータ(@で始まる引数)は使えません。
140         * @og.rev 6.4.1.1 (2016/01/16) StringUtil#csv2Array(String,char) → StringUtil#csv2Array(String,char,int) で、配列長を固定化する。
141         *
142         * @param   value パラメータ(引数は、"DATE.KEY AA BB" などという状態)
143         *
144         * @return   メッセージ情報
145         * @og.rtnNotNull
146         * @see         org.opengion.fukurou.util.HybsDateUtil#getDateFormat( String,String,String,int )
147         * @see         org.opengion.hayabusa.taglib.CommonTagSupport#getDateFormat( String )
148         */
149        private String getDateFormat( final String value ) {
150                // 6.4.1.1 (2016/01/16) StringUtil#csv2Array(String,char) → StringUtil#csv2Array(String,char,int) で、配列長を固定化する。
151                // 足りない分は、null ではなく、"" がセットされてくる。
152                final String[] vals = StringUtil.csv2Array( value,' ',4 );
153
154                // 5.7.4.2 (2014/03/20) 現時点では、SystemParameter 処理にはリクエスト変数は使えないので、@変数も使えない。
155                // vals[0] は、key なので、vals[1] から回す。
156                for( int i=1; i<vals.length; i++ ) {
157                        if( !vals[i].isEmpty() && vals[i].charAt(0) == '@' ) {
158                                final String errMsg = "第" + i + "引数に、リクエストパラメータ(@で始まる引数)は使えません。value=[" + value + "]" ;
159                                throw new OgRuntimeException( errMsg );
160                        }
161                }
162
163                // 5.7.4.1 (2014/03/14) AA 引数の@解析後のコマンド判定方法を、8ケタ以下から先頭が数字以外に変更します。
164                if( !vals[1].isEmpty() ) {
165                        final char chA = vals[1].charAt(0);
166                        if( chA < '0' || chA > '9' ) {          // 先頭が、数字以外の場合は、コマンドなので、一つずつずらす。
167                                vals[3] = vals[2];
168                                vals[2] = vals[1];
169                                vals[1] = null;
170                        }
171                }
172
173                // 5.7.4.1 (2014/03/14) CC(第3引数)を、"H" , "D" , "M" 以外でも使用できるように拡張します。
174                int intC = 0;
175                if( !vals[3].isEmpty() ) {
176                        try {
177                                intC = Integer.parseInt( vals[3] );
178                        }
179                        catch( final NumberFormatException ex ) {
180                                final String errMsg = "第3引数が数字ではありません。value=[" + value + "]"
181                                                                + ex.getMessage() ;
182                                System.err.println( errMsg );
183                        }
184                }
185
186                // AA(第1引数) が null か、isEmpty() の場合は、現在時刻が使用される。
187                return HybsDateUtil.getDateFormat( vals[0],vals[1],vals[2],intC );      // 5.7.4.1 (2014/03/14) CC第3引数を拡張します。
188        }
189
190        /**
191         *  {&#064;XXXX} の特殊文字を含む文字列を、置き換えます。
192         * 対象外の文字列は、そのまま、残されます。
193         *
194         * @og.rev 5.3.4.0 (2011/04/01) 判定方法 修正
195         *
196         * @param       entry   置換文字列のキーと値のペアを管理しているEntryオブジェクトの配列(可変長引数)
197         *
198         * @return      置換後の文字列
199         */
200        public String replace( final HybsEntry... entry ) {
201                // 6.1.1.0 (2015/01/17) 可変長引数でもnullは来る。
202                if( formats == null || entry == null || entry.length == 0 ) { return original; }
203
204                // HybsEntry[] データを、Mapにマッピングします。
205                final Map<String, String> sysMap = new HashMap<>();
206                final int size = entry.length;
207                for( int i=0; i<size; i++ ) {
208                        sysMap.put( entry[i].getKey(),entry[i].getValue() );
209                }
210
211                return replace( sysMap );
212        }
213
214        /**
215         *  {&#064;XXXX} の特殊文字を含む文字列を、置き換えます。
216         * 対象外の文字列は、そのまま、残されます。
217         *
218         * @param  map  置換文字列のキーと値のペアを管理しているMapオブジェクト
219         *
220         * @return      置換後の文字列
221         */
222        public String replace( final Map<String,String> map ) {
223                // 8.5.5.1 (2024/02/29) PMD 7.0.0 OnlyOneReturn メソッドには終了ポイントが 1 つだけ必要
224                if( formats == null || map == null || map.isEmpty() ) { return original; }
225//              if( formats == null ) { return original; }              // 5.3.4.0 (2011/04/01) 判定方法 修正
226//              if( map == null || map.isEmpty() ) { return original; }
227
228                final StringBuilder sb = new StringBuilder( BUFFER_MIDDLE );
229                for( int i=0; i<formats.length; i++ ) {
230                        sb.append( formats[i] );
231                        if( i < clms.length && clms[i] != null ) {              // 5.3.4.0 (2011/04/01) nullチェック追加
232                                sb.append( StringUtil.nval( map.get( clms[i] ), "" ) );
233                        }
234                }
235
236                return sb.toString();
237        }
238
239        /**
240         * フォーマットをパースした結果から、カラム一覧を配列形式で返します。
241         *
242         * @og.rev 5.1.7.0 (2010/06/01) 新規作成
243         *
244         * @return カラム配列
245         * @og.rtnNotNull
246         */
247        public String[] getColumns() {
248                // 6.4.1.1 (2016/01/16) PMD refactoring. A method should have only one exit point, and that should be the last statement in the method
249                return clms == null ? new String[0] : clms.clone();
250        }
251
252        /**
253         * フォーマットをパースした結果から、フォーマット一覧を配列形式で返します。
254         *
255         * @og.rev 5.1.7.0 (2010/06/01) 新規作成
256         *
257         * @return フォーマット配列
258         * @og.rtnNotNull
259         */
260        public String[] getFormats() {
261                // 6.4.1.1 (2016/01/16) PMD refactoring. A method should have only one exit point, and that should be the last statement in the method
262                return formats == null ? new String[0] : formats.clone();
263        }
264}