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.io.FileInputStream; 019import java.io.FileOutputStream; 020 021/** 022 * DateSet.java は、入力ファイルの日付,時刻キーワードを実行時の日時で変換して,出力します。 023 * 024 * 変換には,$(yyyy)の形式で指定し,カッコの文字列は,java.text.SimpleDateFormat で使用する, 025 * 時刻フォーマット構文を用います。 026 * また、引数に keys,vals を渡すことで、$(KEY1) 文字列を VAL1 文字列と置き換えます。 027 * 028 * サンプルファイル 029 * $(yyyy/MM/dd) 年/月/日を表します。 030 * $(yy) 年だけを2桁で表します。 031 * $(MM) 月を2桁 (02,03など)で表します。 032 * $(dd) 日を2桁 (02,03など)で表します。 033 * $(HH:mm:ss) 時:分:秒を表します。 034 * $(MMMMMMMM) 月をフルスペルで表します。 035 * $(MMM) 月を3桁固定(Mar,Aplなど)で表します。 036 * $(EEEEEEEE) 曜日をフルスペルで表します。 037 * $(EEE) 曜日を3桁固定(Sun,Monなど)で表します。 038 * 039 * 時刻フォーマット構文 040 * 041 * 記号 意味 表示 例 042 * ------ ------- ------------ ------- 043 * G 年号 (テキスト) AD 044 * y 年 (数値) 1996 045 * M 月 (テキスト & 数値) July & 07 046 * d 日 (数値) 10 047 * h 午前/午後の時 (1~12) (数値) 12 048 * H 一日における時 (0~23) (数値) 0 049 * m 分 (数値) 30 050 * s 秒 (数値) 55 051 * S ミリ秒 (数値) 978 052 * E 曜日 (テキスト) 火曜日 053 * D 年における日 (数値) 189 054 * F 月における曜日 (数値) 2 (7月の第2水曜日) 055 * w 年における週 (数値) 27 056 * W 月における週 (数値) 2 057 * a 午前/午後 (テキスト) PM 058 * k 一日における時 (1~24) (数値) 24 059 * K 午前/午後の時 (0~11) (数値) 0 060 * z 時間帯 (テキスト) PDT 061 * ' テキスト用エスケープ 062 * '' 単一引用符 ' 063 * 064 * パターン文字のカウントによって、そのフォーマットが決まります。 065 * (テキスト): 4以上: フル形式を使用します。4以下: 短いまたは省力された形式があれば、それを使用します。 066 * 067 * (数値): 最小桁数。これより短い数値は、この桁数までゼロが追加されます。年には特別な処理があります。 068 * つまり、'y'のカウントが2なら、年は2桁に短縮されます。 069 * 070 * (テキスト & 数値): 3以上ならテキストを、それ以外なら数値を使用します。 071 * 072 * パターンの文字が['a'..'z']と['A'..'Z']の範囲になければ、その文字は引用テキストとして扱われます。 073 * たとえば、':'、'.'、' '、'#'、'@'などの文字は、単一引用符に囲まれていなくても、 074 * 結果の時刻テキストに使用されます。 075 * 076 * 無効なパターン文字がパターンに入っていると、フォーマットや解析で例外がスローされます。 077 * 078 * USロケールを使った例: 079 * 080 * フォーマットパターン 結果 081 * -------------------- ---- 082 * "yyyy.MM.dd G 'at' hh:mm:ss z" ⇒ 1996.07.10 AD at 15:08:56 PDT 083 * "EEE, MMM d, ''yy" ⇒ Wed, July 10, '96 084 * "h:mm a" ⇒ 12:08 PM 085 * "hh 'o''''clock' a, zzzz" ⇒ 12 o'clock PM, Pacific Daylight Time 086 * "K:mm a, z" ⇒ 0:00 PM, PST 087 * "yyyyy.MMMMM.dd GGG hh:mm aaa" ⇒ 1996.July.10 AD 12:08 PM 088 * 089 * @version 0.9.0 1999/03/09 090 * @author Kazuhiko Hasegawa 091 * @since JDK1.1, 092 */ 093public class DateSet { 094 private String[] keys = null; 095 private String[] vals = null; 096 097 /** 098 * フォーマット解析時に置き換える キーと値の配列を設定します。 099 * 100 * $(KEY1) 文字列を VAL1 文字列と置き換える処理を行います。これにより日付以外の 101 * 文字列を置き換える処理を実行できます。 102 * 103 * @param inkeys 置き換え元キー配列 104 * @param invals 置き換え元値配列 105 */ 106 public void setKeysVals( final String[] inkeys, final String[] invals ) { 107 if( inkeys != null && invals != null && inkeys.length == invals.length ) { 108 int size = inkeys.length ; 109 keys = new String[size]; 110 vals = new String[size]; 111 System.arraycopy( inkeys,0,keys,0,size ); 112 System.arraycopy( invals,0,vals,0,size ); 113 } 114 } 115 116 /** 117 * 現在日付、時刻をフォーマット指定個所に埋め込みます。 118 * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。 119 * 120 * @param inByte 変換元バイト配列 121 * 122 * @return 変換後のバイト配列 123 */ 124 public byte[] change( final byte[] inByte ) { 125 byte[] outByte = new byte[ inByte.length+100 ]; 126 int add = 0; 127 for( int i=0; i<inByte.length; i++) { 128 if( inByte[i] == '$' && i<inByte.length-1 && inByte[i+1] == '(' ) { 129 int j = 0; 130 while( inByte[i+j+2] != ')') { j++; } 131 String str = changeForm( new String( inByte,i+2,j,StringUtil.DEFAULT_CHARSET ) ); // 5.5.2.6 (2012/05/25) findbugs対応 132 byte[] byteDate = str.getBytes( StringUtil.DEFAULT_CHARSET ) ; // 5.5.2.6 (2012/05/25) findbugs対応 133 for( int k = 0; k<byteDate.length; k++) { 134 outByte[add] = byteDate[k]; 135 add++; 136 } 137 i += j+2; 138 } 139 else { 140 outByte[add] = inByte[i]; 141 add++; 142 } 143 } 144 byte[] rtnByte = new byte[ add ]; 145 System.arraycopy( outByte,0,rtnByte,0,add ); 146 return rtnByte; 147 } 148 149 /** 150 * keys,vals の変換、および、現在日付、時刻のフォーマット変換を行います。 151 * 152 * 先に、keys,vals の変換を行います。form が、keys にマッチすれば、vals を 153 * 返します。最後までマッチしなければ、時刻のフォーマット変換を行います。 154 * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。 155 * 156 * @og.rev 5.5.7.2 (2012/10/09) HybsDateUtil を利用するように修正します。 157 * 158 * @param form フォーム文字列 ( 例 "yyyy/MM/dd HH:mm:ss" ) 159 * 160 * @return フォーマット変換結果 161 */ 162 public String changeForm( final String form ) { 163 if( keys != null ) { 164 for( int i=0; i<keys.length; i++ ) { 165 if( form.equals( keys[i] ) ) { 166 return vals[i]; 167 } 168 } 169 } 170 171 return HybsDateUtil.getDate( form ); 172 } 173 174 /** 175 * keys,vals の変換、および、現在日付、時刻のフォーマット変換を行います。 176 * 177 * 先に、keys,vals の変換を行います。form が、keys にマッチすれば、vals を 178 * 返します。最後までマッチしなければ、時刻のフォーマット変換を行います。 179 * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。 180 * 181 * @param form フォーム文字列 ( 例 "yyyy/MM/dd HH:mm:ss" ) 182 * 183 * @return フォーマット変換結果 184 */ 185 public String changeString( final String form ) { 186 StringBuilder buf = new StringBuilder( 200 ); 187 int bkst = 0; 188 int st = form.indexOf( "$(" ); 189 while( st >= 0 ) { 190 buf.append( form.substring( bkst,st ) ); 191 int ed = form.indexOf( ")",st+2 ); 192 buf.append( changeForm( form.substring( st+2,ed ) ) ); 193 bkst = ed + 1; 194 st = form.indexOf( "$(",bkst ); 195 } 196 buf.append( form.substring( bkst ) ); 197 198 return buf.toString(); 199 } 200 201 /** 202 * 入力ファイルの時刻フォーマットを変換して出力ファイルに書き込みます。 203 * 204 * 引数に <key1> <val1> のペア情報を渡すことが可能です。 205 * 先に、keys,vals の変換を行います。form が、keys にマッチすれば、vals を 206 * 返します。最後までマッチしなければ、時刻のフォーマット変換を行います。 207 * フォーマットの指定方法は、java.text.SimpleDateFormat の指定方法と同一です。 208 * フォーム文字列例 ( "yyyy/MM/dd HH:mm:ss" ) 209 * 210 * @param args 引数配列( 入力ファイル 出力ファイル キー1 値1 ・・・ 211 * @throws Throwable なんらかのエラーが発生した場合。 212 */ 213 public static void main( final String[] args ) throws Throwable { 214 if( args.length > 2 && ( args.length % 2 != 0 ) ) { 215 System.err.println( "Usage: java org.opengion.fukurou.util.DateSet <inputFile> <outputFile> [<key1> <val1> ・・・]" ); 216 return ; 217 } 218 219 String[] keys = new String[ (args.length-2)/2 ]; 220 String[] vals = new String[ (args.length-2)/2 ]; 221 for( int i=1; i<=keys.length; i++ ) { 222 keys[i-1] = args[i*2]; 223 vals[i-1] = args[i*2+1]; 224 } 225 226 FileInputStream filein = new FileInputStream( args[0] ); 227 byte[] byteIn = new byte[ filein.available() ]; 228 int len = filein.read( byteIn ); 229 if( len != byteIn.length ) { 230 String errMsg = "読み取りファイルのデータが切り捨てられました。" + 231 "File=" + args[0] + " Length=" + len + " Input=" + byteIn.length ; 232 System.err.println( errMsg ); 233 } 234 filein.close(); 235 236 DateSet dateSet = new DateSet(); 237 dateSet.setKeysVals( keys,vals ); 238 byte[] byteout = dateSet.change( byteIn ); 239 240 FileOutputStream fileout = new FileOutputStream( args[1] ); 241 fileout.write( byteout ); 242 fileout.close(); 243 } 244}