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.system;                                                            // 6.4.2.0 (2016/01/29) package変更 fukurou.util → fukurou.system
017
018import java.io.FileInputStream;
019import java.io.FileOutputStream;
020import java.util.Date;
021import java.util.Locale;
022// import java.util.Calendar;                                                                                   // 7.0.1.4 (2018/11/26)
023import java.util.concurrent.ConcurrentMap;                                                      // 7.0.1.3 (2018/11/12)
024import java.util.concurrent.ConcurrentHashMap;                                          // 7.0.1.3 (2018/11/12)
025import java.text.DateFormat;
026import java.text.SimpleDateFormat;
027
028// import static org.opengion.fukurou.system.HybsConst.BUFFER_MIDDLE;   // 6.1.0.0 (2014/12/26) refactoring
029
030/**
031 * DateSet.java は、入力ファイルの日付、時刻キーワードを実行時の日時で変換して出力します。
032 *
033 * 変換には、$(yyyy)の形式で指定し、カッコの文字列は java.text.SimpleDateFormat で使用する
034 * 時刻フォーマット構文を用います。
035 * また、引数に keys,vals を渡すことで、$(KEY1) 文字列を VAL1 文字列と置き換えます。
036 *
037 *  サンプルファイル
038 *  $(yyyy/MM/dd)        年/月/日を表します。
039 *  $(yy)                年だけを2桁で表します。
040 *  $(MM)                月を2桁 (02,03など)で表します。
041 *  $(dd)                日を2桁 (02,03など)で表します。
042 *  $(HH:mm:ss)          時:分:秒を表します。
043 *  $(MMMMMMMM)          月をフルスペルで表します。
044 *  $(MMM)               月を3桁固定(Mar,Aplなど)で表します。
045 *  $(EEEEEEEE)          曜日をフルスペルで表します。
046 *  $(EEE)               曜日を3桁固定(Sun,Monなど)で表します。
047 *
048 *  // 7.0.1.3 (2018/11/12)                                                      2019/01/01    2030/01/01
049 *  $(ATIME)             通算秒数( new Date().getTime()/1000 )  の 10桁文字列    1546268400    1893423600
050 *  $(BTIME)             通算分数( new Date().getTime()/60000 ) の  8桁文字列      25771140      31557060
051 *    ※ BTIME が桁あふれするのは、2160/02/18 です。
052 *
053 *   時刻フォーマット構文
054 *
055 *   記号     意味                    表示                例
056 *   ------   -------                 ------------        -------
057 *   G        年号                    (テキスト)          AD
058 *   y        年                      (数値)              1996
059 *   M        月                      (テキスト & 数値)  July & 07
060 *   d        日                      (数値)              10
061 *   h        午前/午後の時 (1~12)    (数値)              12
062 *   H        一日における時 (0~23)   (数値)              0
063 *   m        分                      (数値)              30
064 *   s        秒                      (数値)              55
065 *   S        ミリ秒                  (数値)              978
066 *   E        曜日                    (テキスト)          火曜日
067 *   D        年における日            (数値)              189
068 *   F        月における曜日          (数値)              2 (7月の第2水曜日)
069 *   w        年における週            (数値)              27
070 *   W        月における週            (数値)              2
071 *   a        午前/午後               (テキスト)          PM
072 *   k        一日における時 (1~24)   (数値)              24
073 *   K        午前/午後の時 (0~11)    (数値)              0
074 *   z        時間帯                  (テキスト)          PDT
075 *   '        テキスト用エスケープ
076 *   ''       単一引用符                                  '
077 *
078 *  パターン文字のカウントによって、そのフォーマットが決まります。
079 *  (テキスト): 4以上: フル形式を使用します。4以下: 短いまたは省力された形式があれば、それを使用します。
080 *
081 *  (数値): 最小桁数。これより短い数値は、この桁数までゼロが追加されます。年には特別な処理があります。
082 *  つまり、'y'のカウントが2なら、年は2桁に短縮されます。
083 *
084 *  (テキスト & 数値): 3以上ならテキストを、それ以外なら数値を使用します。
085 *
086 *  パターンの文字が['a'..'z']と['A'..'Z']の範囲になければ、その文字は引用テキストとして扱われます。
087 *  たとえば、':'、'.'、' '、'#'、'@'などの文字は、単一引用符に囲まれていなくても、
088 *  結果の時刻テキストに使用されます。
089 *
090 *  無効なパターン文字がパターンに入っていると、フォーマットや解析で例外がスローされます。
091 *
092 *  USロケールを使った例:
093 *
094 *   フォーマットパターン                   結果
095 *   --------------------                   ----
096 *   "yyyy.MM.dd G 'at' hh:mm:ss z"    ⇒  1996.07.10 AD at 15:08:56 PDT
097 *   "EEE, MMM d, ''yy"                ⇒  Wed, July 10, '96
098 *   "h:mm a"                          ⇒  12:08 PM
099 *   "hh 'o''''clock' a, zzzz"         ⇒  12 o'clock PM, Pacific Daylight Time
100 *   "K:mm a, z"                       ⇒  0:00 PM, PST
101 *   "yyyyy.MMMMM.dd GGG hh:mm aaa"    ⇒  1996.July.10 AD 12:08 PM
102 *
103 * @og.rev 6.4.2.0 (2016/01/29) package変更 fukurou.util → fukurou.system
104 * @og.rev 7.0.1.3 (2018/11/12) keys,valsをMapに変更。BUILD_TYPEのビルド番号を100秒単位に変更。
105 *
106 * @version  0.9.0  1999/03/09
107 * @author   Kazuhiko Hasegawa
108 * @since    JDK1.1,
109 */
110public class DateSet {
111        private final ConcurrentMap<String,String> prmMap = new ConcurrentHashMap<>();          // 7.0.1.3 (2018/11/12)
112
113//      private String[] keys ;
114//      private String[] vals ;
115
116        /**
117         * デフォルトコンストラクター
118         *
119         * Mapの初期値を設定しておきます。
120         *
121         * @og.rev 7.0.1.3 (2018/11/12) KeysValsは、Mapに置き換え
122         * @og.rev 7.0.1.4 (2018/11/26) ATIME,BTIME 追加
123         */
124        public DateSet() {
125                prmMap.put( "ATIME" ,  Long.toString( new Date().getTime()/1000 ) );            // 
126                prmMap.put( "BTIME" ,  Long.toString( new Date().getTime()/60000 ) );           // 
127
128        //      final Calendar now = Calendar.getInstance();
129        //      final int tm = ( now.get( Calendar.HOUR_OF_DAY ) * 60 + now.get( Calendar.MINUTE ) ) / 15 ;             // 0 ~ 96未満の数値になる
130
131        //      final String CTIME = getDate( "yyDDD" ) + String.format( "%02d" , tm ) ;
132        //      prmMap.put( "CTIME" ,  CTIME );
133        }
134
135        /**
136         * フォーマット解析時に置き換える キーと値のMapを設定します。
137         *
138         * $(KEY1) 文字列を VAL1 文字列と置き換える処理を行います。これにより日付以外の
139         * 文字列を置き換える処理を実行できます。
140         *
141         * @og.rev 7.0.1.3 (2018/11/12) KeysValsは、Mapに置き換え
142         *
143         * @param       pMap    置き換え元のMap
144         */
145        public void setParamMap( final ConcurrentMap<String,String> pMap ) {
146                if( pMap != null ) {
147                         prmMap.putAll( pMap );
148                }
149        }
150
151//      /**
152//       * フォーマット解析時に置き換える キーと値の配列を設定します。
153//       *
154//       * $(KEY1) 文字列を VAL1 文字列と置き換える処理を行います。これにより日付以外の
155//       * 文字列を置き換える処理を実行できます。
156//       *
157//       * @og.rev 7.0.1.3 (2018/11/12) KeysValsは、Mapに置き換え
158//       *
159//       * @param       inkeys  置き換え元キー配列
160//       * @param       invals  置き換え元値配列
161//       */
162//      public void setKeysVals( final String[] inkeys, final String[] invals ) {
163//              if( inkeys != null && invals != null && inkeys.length == invals.length ) {
164//                      final int size = inkeys.length ;
165//                      keys = new String[size];
166//                      vals = new String[size];
167//                      System.arraycopy( inkeys,0,keys,0,size );
168//                      System.arraycopy( invals,0,vals,0,size );
169//              }
170//      }
171
172        /**
173         * 現在日付、時刻をフォーマット指定個所に埋め込みます。
174         * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。
175         *
176         * @og.rev 6.3.6.0 (2015/08/16) System.arraycopy が使える箇所は、置き換えます。
177         * @og.rev 6.4.2.0 (2016/01/29) fukurou.util.StringUtil → fukurou.system.HybsConst に変更
178         *
179         * @param       inByte  変換元バイト配列(可変長引数)
180         * @return      変換後のバイト配列
181         */
182        public byte[] change( final byte... inByte ) {
183                byte[] outByte = new byte[inByte.length+100];
184                int add = 0;
185                for( int i=0; i<inByte.length; i++ ) {
186                        if( inByte[i] == '$' && i<inByte.length-1 && inByte[i+1] == '(' ) {
187                                int j = 0;
188                                while( inByte[i+j+2] != ')') { j++; }
189                                final String str = changeForm( new String( inByte,i+2,j,HybsConst.DEFAULT_CHARSET ) );          // 6.4.2.0 (2016/01/29)
190                                final byte[] byteDate = str.getBytes( HybsConst.DEFAULT_CHARSET ) ;                                                     // 6.4.2.0 (2016/01/29)
191                                // 6.3.6.0 (2015/08/16) System.arraycopy が使える箇所は、置き換えます。
192                                System.arraycopy( byteDate,0,outByte,add,byteDate.length );             // 6.3.6.0 (2015/08/16)
193                                add += byteDate.length ;                                                                                // 6.3.6.0 (2015/08/16)
194                                i   += j+2;
195                        }
196                        else {
197                                outByte[add] = inByte[i];
198                                add++;
199                        }
200                }
201                final byte[] rtnByte = new byte[add];
202                System.arraycopy( outByte,0,rtnByte,0,add );
203                return rtnByte;
204        }
205
206        /**
207         * パラメータの変換、および、現在日付、時刻のフォーマット変換を行います。
208         *
209         * 先に、パラメータの変換を行います。form が、Mapのkey にマッチすれば、
210         * その値を返します。マッチしなければ、時刻のフォーマット変換を行います。
211         * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。
212         *
213         * @og.rev 5.5.7.2 (2012/10/09) HybsDateUtil を利用するように修正します。
214         * @og.rev 6.3.9.0 (2015/11/06) コンストラクタで初期化されていないフィールドを null チェックなしで利用している(findbugs)
215         * @og.rev 7.0.1.3 (2018/11/12) KeysValsは、Mapに置き換え
216         *
217         * @param       form フォーム文字列 ( 例 "yyyy/MM/dd HH:mm:ss" )
218         *
219         * @return      フォーマット変換結果
220         */
221        public String changeForm( final String form ) {
222                // Map#getOrDefault( key,defVal )はdefValも評価が必ず実行されるため、使えない。
223                // return prmMap.getOrDefault( form,DateSet.getDate( form ) ;
224
225                final String rtn = prmMap.get( form );
226
227                return rtn == null ? DateSet.getDate( form ) : rtn ;
228
229//              // 6.3.9.0 (2015/11/06) コンストラクタで初期化されていないフィールドを null チェックなしで利用している(findbugs)
230//              if( keys != null && vals != null ) {
231//                      for( int i=0; i<keys.length; i++ ) {
232//                              if( form.equals( keys[i] ) ) {
233//                                      return vals[i];
234//                              }
235//                      }
236//              }
237//
238//              return DateSet.getDate( form );
239        }
240
241        /**
242         * パラメータの変換、および、現在日付、時刻のフォーマット変換を行います。
243         *
244         * 先に、パラメータの変換を行います。form が、Mapのkey にマッチすれば、
245         * その値を返します。マッチしなければ、時刻のフォーマット変換を行います。
246         * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。
247         *
248         * @param       form フォーム文字列 ( 例 "yyyy/MM/dd HH:mm:ss" )
249         *
250         * @return      フォーマット変換結果
251         * @og.rtnNotNull
252         */
253        public String changeString( final String form ) {
254                final StringBuilder buf = new StringBuilder( HybsConst.BUFFER_MIDDLE );
255                int bkst = 0;
256                int st = form.indexOf( "$(" );
257                while( st >= 0 ) {
258                        buf.append( form.substring( bkst,st ) );
259                        final int ed = form.indexOf( ')',st+2 );                                // 6.0.2.5 (2014/10/31) refactoring
260                        buf.append( changeForm( form.substring( st+2,ed ) ) );
261                        bkst = ed + 1;
262                        st = form.indexOf( "$(",bkst );
263                }
264                buf.append( form.substring( bkst ) );
265
266                return buf.toString();
267        }
268
269        /**
270         * 現在日付、時刻を指定のフォーマットで文字列に変換して返します。
271         * 出力フォーマットは、"yyyy/MM/dd HH:mm:ss" 固定です。
272         *
273         * @og.rev 5.5.7.2 (2012/10/09) 新規作成
274         * @og.rev 6.4.2.0 (2016/01/29) fukurou.util.HybsDateUtil → fukurou.system.DateSet に変更
275         *
276         * @return      現在日付、時刻 ( 例 2012/09/05 18:10:24 )
277         * @og.rtnNotNull
278         */
279        public static String getDate() {
280                final DateFormat formatter = new SimpleDateFormat( "yyyy/MM/dd HH:mm:ss",Locale.JAPAN );
281                return formatter.format( new Date() );
282        }
283
284        /**
285         * 現在時刻を指定のフォーマットで文字列に変換して返します。
286         * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。
287         * 変換時のロケーションは、Locale.JAPAN です。
288         * 現在時刻は、new Date() で求めます。
289         *
290         * @param       form フォーム文字列 ( 例 "yyyy/MM/dd HH:mm:ss.SSS" )
291         *
292         * @og.rev 5.5.7.2 (2012/10/09) 新規作成
293         * @og.rev 6.4.2.0 (2016/01/29) fukurou.util.HybsDateUtil → fukurou.system.DateSet に変更
294         *
295         * @return      現在日付、時刻
296         * @og.rtnNotNull
297         * @see         java.text.SimpleDateFormat
298         */
299        public static String getDate( final String form ) {
300                final DateFormat formatter = new SimpleDateFormat( form,Locale.JAPAN );
301                return formatter.format( new Date() );
302        }
303
304        /**
305         * 指定時刻を指定のフォーマットで文字列に変換して返します。
306         * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。
307         * 変換時のロケーションは、Locale.JAPAN です。
308         * 指定時刻は、new Date( time ) で求めます。
309         *
310         * @param       time 指定のカレントタイムのロング値
311         * @param       form フォーム文字列 ( 例 "yyyy/MM/dd HH:mm:ss.SSS" )
312         *
313         * @og.rev 5.5.7.2 (2012/10/09) 新規作成
314         * @og.rev 6.4.2.0 (2016/01/29) fukurou.util.HybsDateUtil → fukurou.system.DateSet に変更
315         *
316         * @return      現在日付、時刻( 例 2001/04/17 15:48:22 )
317         * @og.rtnNotNull
318         */
319        public static String getDate( final long time,final String form ) {
320                final DateFormat formatter = new SimpleDateFormat( form,Locale.JAPAN );
321                return formatter.format( new Date( time ) );
322        }
323
324        /**
325         * 入力ファイルの時刻フォーマットを変換して出力ファイルに書き込みます。
326         *
327         * 引数に &lt;key1&gt; &lt;val1&gt; のペア情報を渡すことが可能です。
328         * 先に、keys,vals の変換を行います。form が、keys にマッチすれば、vals を
329         * 返します。最後までマッチしなければ、時刻のフォーマット変換を行います。
330         * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。
331         * フォーム文字列例 (  "yyyy/MM/dd HH:mm:ss" )
332         *
333         * @og.rev 7.0.1.3 (2018/11/12) KeysValsは、Mapに置き換え
334         *
335         * @param       args 引数配列( 入力ファイル 出力ファイル キー1 値1 ・・・
336         * @throws Throwable なんらかのエラーが発生した場合。
337         */
338        public static void main( final String[] args ) throws Throwable {
339                if( args.length > 2 && ( args.length % 2 != 0 ) ) {
340                        System.err.println( "Usage: java org.opengion.fukurou.system.DateSet <inputFile> <outputFile> [<key1> <val1> ・・・]" );
341                        return ;
342                }
343
344                final ConcurrentMap<String,String> prmMap = new ConcurrentHashMap<>();          // 7.0.1.3 (2018/11/12)
345
346                for( int i=2; i<args.length; i+=2 ) {
347                        prmMap.put( args[i] , args[i+1] );
348                }
349
350//              String[] keys = new String[ (args.length-2)/2 ];
351//              String[] vals = new String[ (args.length-2)/2 ];
352//              for( int i=1; i<=keys.length; i++ ) {
353//                      keys[i-1] = args[i*2];
354//                      vals[i-1] = args[i*2+1];
355//              }
356
357                final FileInputStream filein = new FileInputStream( args[0] );
358                final byte[] byteIn = new byte[ filein.available() ];
359                final int len = filein.read( byteIn );
360                if( len != byteIn.length ) {
361                        final String errMsg = "読み取りファイルのデータが切り捨てられました。" +
362                                                        "File=" + args[0] + " Length=" + len  + " Input=" + byteIn.length ;
363                        System.err.println( errMsg );
364                }
365                filein.close();
366
367                final DateSet dateSet = new DateSet();
368//              dateSet.setKeysVals( keys,vals );
369                dateSet.setParamMap( prmMap );
370                final byte[] byteout = dateSet.change( byteIn );
371
372                final FileOutputStream fileout = new FileOutputStream( args[1] );
373                fileout.write( byteout );
374                fileout.close();
375        }
376}