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.hayabusa.taglib; 017 018import static org.opengion.fukurou.util.StringUtil.*; 019 020import java.io.File; 021import java.io.IOException; 022import java.util.ArrayList; 023import java.util.List; 024 025import javax.servlet.http.HttpServletRequest; 026 027import org.opengion.fukurou.util.StringUtil; 028import org.opengion.hayabusa.common.HybsSystem; 029import org.opengion.hayabusa.common.HybsSystemException; 030import org.opengion.hayabusa.db.DBColumn; 031import org.opengion.hayabusa.db.DBTableModel; 032import org.opengion.hayabusa.db.DBTableModelUtil; 033import org.opengion.hayabusa.servlet.MultipartRequest; 034import org.opengion.hayabusa.servlet.UploadedFile; 035 036/** 037 * クライアントのファイルをサーバーにアップロードするタグです。 038 * 039 * アップロード後の属性は、DBTableModel に格納することにより、通常のデータと 040 * 同様の取り出し方が可能です。 041 * また、通常のファイルアップロード時の、form で使用する、enctype="multipart/form-data" 042 * を指定した場合の、他のリクエスト情報も、{@XXXX} 変数で取り出すことが可能です。 043 * 044 * この upload タグでは、アップロード後に、指定のファイル名に変更する機能があります。 045 * file 登録ダイアログで指定した name に、"_NEW" という名称を付けたリクエスト値を 046 * ファイルのアップロードと同時に送信することで、この名前にファイルを付け替えます。 047 * また、アップロード後のファイル名は、name 指定の名称で、取り出せます。 048 * クライアントから登録したオリジナルのファイル名は、name に、"_ORG" という名称 049 * で取り出すことが可能です。 050 * 051 * 通常では、これらのパラメータを、RequestAttribute 変数にセットしますので、{@XXXX}で 052 * 取り出すことが可能になります。さらに、"KEY","VALUE","ISFILE" のカラムを持った、 053 * DBTableModel にセットします。 054 * 055 * 新機能として、columns を指定する事で、columns のカラムを持つ DBTableModel にセットします。 056 * その場合は、カラム名_01 〜 カラム名_99 のように、アンダーバーで列データとなるキーを定義してください。 057 * アンダーバーがない場合は、カラムだけ作成されます。カラム名と同じリクエストがあれば、 058 * すべてのレコードに同じ値がセットされます。 059 * 060 * 新ファイル名に拡張子が設定されていない場合は、オリジナルファイル名の拡張子をセットします。 061 * 062 * HTML5 の type="file" 時の multiple 属性(アップロードファイルの複数選択機能)に対応します。(5.7.1.1 (2013/12/13)) 063 * その場合は、新しいファイル名への変更はできません。オリジナルのファイル名でアップロードされます 064 * 065 * 5.7.1.2 (2013/12/20) zip 対応 066 * filename 属性に、".zip" の拡張子のファイル名を指定した場合は、アップロードされた一連のファイルを 067 * ZIP圧縮します。これは、アップロード後の処理になります。 068 * ZIP圧縮のオリジナルファイルは、そのまま残ります。 069 * なお、ZIPファイルは、useBackup属性を true に設定しても、無関係に、上書きされます。 070 * 071 * 5.7.4.3 (2014/03/28) filename 属性のリクエスト変数対応 072 * filename 属性のみ、{@XXXX} のリクエスト変数が使えるようにします。 073 * 他のパラメータでは使えません。 074 * これは、multipart/form-data のリクエストでは、パートの分解処理をしないと、リクエスト変数が 075 * 拾えない為、リクエスト変数は、この、upload タグ以降でのみ利用可能でした。 076 * zip対応と関連付けて、filename 属性のみ、利用できるように、MultipartRequest 側の処理に組み込みます。 077 * 078 * 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 079 * 個々に指定したアップロードファイル名は、XXX_NEW や XXX_ORG で取得できますが、 080 * HTML5 の multiple 属性使用時や、アップロードされたファイルを一連で処理したい場合に 081 * ファイル名を、CSV形式で取り出せるようにします。 082 * キーは、共通で、UPLOAD_FILES とします。 083 * 084 * 5.9.25.0 (2017/10/06) 085 * クラウド上のPaaSでオブジェクトストレージを利用する際は以下のシステムリソースを設定してください。 086 * CLOUD_STORAGE,CLOUD_STORAGE_CONTAINER 087 * plugin/cloud内のクラスを利用してファイルアップロード(FileUploadタグ)、ダウンロード(FileDownloadサーブレット)をAPI経由で行います。 088 * プラグインが利用するjarファイルの配置は必要です。 089 * 090 * @og.formSample 091 * ●形式:<og:upload fileURL="…" maxPostSize="…" /> 092 * ●body:なし 093 * 094 * ●Tag定義: 095 * <og:upload 096 * fileURL 【TAG】ファイルをアップロードするディレクトリを指定します (初期値:FILE_URL[=filetemp/]) 097 * filename 【TAG】(通常使いません)ファイルを作成するときのファイル名をセットします 098 * maxPostSize 【TAG】最大転送サイズ(Byte)を指定します(初期値:10485760) 0,またはマイナスで無制限です。 099 * scope 【TAG】キャッシュする場合のスコープ[request/page/session/applicaton]を指定します(初期値:session) 100 * tableId 【TAG】(通常使いません)sessionから所得する DBTableModelオブジェクトの ID 101 * columns 【TAG】DBTableModel作成時に、指定のカラムの"_01"〜"_99"の添え字をレコードとして作成します。 102 * useBackup 【TAG】ファイルアップロード時に、すでに同名のファイルが存在した場合に、バックアップ処理(renameTo)するかどうか[true/false]を指定します(初期値:false) 103 * language 【TAG】タグ内部で使用する言語コード[ja/en/zh/…]を指定します 104 * storageType 【TAG】保存先ストレージタイプを指定します 105 * bucketName 【TAG】保存先バケット名を指定します 106 * debug 【TAG】デバッグ情報を出力するかどうか[true/false]を指定します(初期値:false) 107 * /> 108 * 109 * ●使用例 : 110 * 【query.jsp】 111 * <form method="POST" action="result.jsp" enctype="multipart/form-data" target="RESULT"> 112 * <table summary="layout" > 113 * <tr><og:input type="text" name="submitter" value="{@USER.JNAME}" size="20" lbl="MSG0014" /></tr> 114 * <tr> 115 * <og:input type="file" name="file_01" size="30" lbl="MSG0015" /> 116 * <og:input name="file_01_NEW" size="10" lbl="FILENAME" /> 117 * </tr><tr> 118 * <og:input type="file" name="file_02" size="30" lbl="MSG0015" /> 119 * <og:input name="file_02_NEW" size="10" lbl="FILENAME" /> 120 * </tr><tr> 121 * <og:input type="file" name="file_03" size="30" lbl="MSG0015" /> 122 * <og:input name="file_03_NEW" size="10" lbl="FILENAME" /> 123 * </tr><tr> 124 * <og:column name="writable" value="false" /> 125 * </tr> 126 * </table> 127 * 128 * 【result.jsp】 129 * <og:upload 130 * fileURL = "{@USER.ID}" 131 * /> 132 * <br /> 133 * <og:message lbl="MSG0003" comment="ファイルの登録が完了しました。" /> 134 * 135 * <og:view 136 * command = "NEW" 137 * viewFormType = "HTMLTable" 138 * writable = "{@writable}" 139 * /> 140 * 141 * <table> 142 * <tr><og:input name="submitter" value="{@submitter}" /></tr> 143 * <tr><og:input name="writable" value="{@writable}" /></tr> 144 * <tr><og:input name="directory" value="{@directory}" /></tr> 145 * <tr><og:input name="file_01" value="{@file_01}" /></tr> 146 * <tr><og:input name="file_01_NEW" value="{@file_01_NEW}" /></tr> 147 * <tr><og:input name="file_01_ORG" value="{@file_01_ORG}" /></tr> 148 * <tr><og:input name="file_02" value="{@file_02}" /></tr> 149 * <tr><og:input name="file_02_NEW" value="{@file_02_NEW}" /></tr> 150 * <tr><og:input name="file_02_ORG" value="{@file_02_ORG}" /></tr> 151 * <tr><og:input name="file_03" value="{@file_03}" /></tr> 152 * <tr><og:input name="file_03_NEW" value="{@file_03_NEW}" /></tr> 153 * <tr><og:input name="file_03_ORG" value="{@file_03_ORG}" /></tr> 154 * </table> 155 * 156 * 【result.jsp】 157 * <og:upload 158 * fileURL = "{@USER.ID}" 159 * columns = "submitter,file" 160 * /> 161 * <br /> 162 * <og:message lbl="MSG0003" comment="ファイルの登録が完了しました。" /> 163 * 164 * <og:view 165 * command = "NEW" 166 * viewFormType = "HTMLTable" 167 * writable = "{@writable}" 168 * /> 169 * 170 * @og.rev 5.10.9.0 (2019/03/01) oota クラウドストレージ対応を追加。(Fileクラスを拡張) 171 * 172 * @og.group ファイル入力 173 * 174 * @version 4.0 175 * @author Kazuhiko Hasegawa 176 * @since JDK5.0, 177 */ 178public class FileUploadTag extends CommonTagSupport { 179 //* このプログラムのVERSION文字列を設定します。 {@value} */ 180 private static final String VERSION = "5.7.6.3 (2014/05/23)" ; 181 182 private static final long serialVersionUID = 576320140523L ; 183 184 /** 5.7.6.3 (2014/05/23) アップロードファイルのCSVセットのキー */ 185 public static final String UPLOAD_FILES = "UPLOAD_FILES" ; 186 187 // 3.5.2.0 (2003/10/20) カラム名に、ISFILEを追加。 188 private static final String[] names = new String[] { "KEY","VALUE","ISFILE" }; 189 private static final String ENCODE = "UTF-8"; // 3.5.2.0 (2003/10/20) 廃止 190 private String fileURL = HybsSystem.sys( "FILE_URL" ); 191 private String filename = null; // 3.5.4.2 (2003/12/15) 192 193 private int maxPostSize = 10*1024*1024; // 最大ファイル容量 10MB 194 private String tableId = HybsSystem.TBL_MDL_KEY ; 195 196 // 5.6.5.2 (2013/06/21) DBTableModel作成時に、指定のカラムの"_01"〜"_99"の添え字をレコードとして作成します。 197 private String columns = null; 198 // 5.6.5.3 (2013/06/28) ファイルアップロード時に、すでに同名のファイルが存在した場合に、バックアップ処理(renameTo)するかどうか[true/false]を指定します(初期値:false) 199 private boolean useBackup = false; 200 private String storageType = null; // 5.10.9.0 (2019/03/01) ADD 201 private String bucketName = null; // 5.10.9.0 (2019/03/01) ADD 202 203 /** 204 * Taglibの終了タグが見つかったときに処理する doEndTag() を オーバーライドします。 205 * 206 * @og.rev 2.2.0.0 (2002/12/17) 中国語(国際化)対応 エンコードの取得方法変更 207 * @og.rev 3.1.1.2 (2003/04/04) Tomcat4.1 対応。release2() を doEndTag()で呼ぶ。 208 * @og.rev 3.1.3.0 (2003/04/10) UTF-8 決め打ちのエンコード情報を取得する。 209 * @og.rev 3.5.2.0 (2003/10/20) scope 属性を追加。 210 * @og.rev 3.5.4.2 (2003/12/15) ファイル名を指定できるようにします。 211 * @og.rev 3.6.0.8 (2004/11/19) DBTableModel をセーブする時に、トランザクションチェックを行います。 212 * @og.rev 3.7.1.1 (2005/05/23) フォルダがない場合は、複数階層分のフォルダを自動で作成します。 213 * @og.rev 3.8.1.3A (2006/01/30) 新ファイル名にオリジナルファイル名の拡張子をセットします 214 * @og.rev 5.3.7.0 (2011/07/01) エラーメッセージ内容変更 215 * @og.rev 5.6.5.2 (2013/06/21) columns 属性の追加 216 * @og.rev 5.6.5.3 (2013/06/28) useBackup 属性の追加 217 * @og.rev 5.8.8.0 (2015/06/05) エラー文の日本語化 218 * @og.rev 5.9.25.0 (2017/10/06) クラウドストレージの利用追加対応 219 * @og.rev 5.10.9.0 (2019/03/01) クラウドストレージ対応の追加 220 * 221 * @return 後続処理の指示 222 */ 223 @Override 224 public int doEndTag() { 225 debugPrint(); // 4.0.0 (2005/02/28) 226 startQueryTransaction( tableId ); // 3.6.0.8 (2004/11/19) 227 HttpServletRequest request = (HttpServletRequest)getRequest(); 228 229 try { 230 String directory = HybsSystem.url2dir( fileURL ); 231 File dir = new File(directory); 232 if( ! dir.exists() && ! dir.mkdirs() ) { 233// String errMsg = "ディレクトリの作成に失敗しました。[" + directory + "]"; 234 String errMsg = getResource().getLabel( "ERR0043" ) + "[" + directory + "]"; // 5.8.8.0 (2015/06/05) 235 throw new HybsSystemException( errMsg ); 236 } 237 238 // 3.8.1.3A (2006/01/30) 新ファイル名にオリジナルファイル名の拡張子をセットします 239// MultipartRequest multi = new MultipartRequest( request,directory,maxPostSize,ENCODE,filename ); 240 // 5.6.5.3 (2013/06/28) useBackup 属性の追加 241 // 5.9.25.0 (2017/10/06) fileURL 属性の追加 242 // 5.10.9.0 (2019/03/01) storageType, bucketName を追加。 243// MultipartRequest multi = new MultipartRequest( request,directory,maxPostSize,ENCODE,filename,useBackup,fileURL ); 244 MultipartRequest multi = new MultipartRequest( request,directory,maxPostSize,ENCODE,filename,useBackup,fileURL,storageType,bucketName ); 245 // 5.6.5.2 (2013/06/21) columns 属性の追加 246// DBTableModel table = makeDBTable( multi ); 247 DBTableModel table = null; 248 if( columns == null ) { // 5.6.5.2 (2013/06/21) columns 属性の追加 249 table = makeDBTable( multi ); 250 } 251 else { 252 table = makeDBTableFromClms( multi ); 253 } 254 255 // 3.5.2.0 (2003/10/20) scope 属性を追加。 256 // 3.6.0.8 (2004/11/19) トランザクションチェックを行います。 257 if( ! commitTableObject( tableId, table ) ) { 258// jspPrint( "FileUploadTag Query処理が割り込まれました。DBTableModel は登録しません。" ); 259 jspPrint( "FileUploadTag " + getResource().getLabel( "ERR0041" ) ); // 5.8.8.0 (2015/06/05) 260 return (SKIP_PAGE); 261 } 262 263 } catch(IOException ex) { 264// String errMsg = "ファイルの取り扱い中にエラーが発生しました。" 265// + toString() + HybsSystem.CR 266// + "FileURL=" + fileURL + HybsSystem.CR 267// + ex.getMessage(); // 5.1.8.0 (2010/07/01) errMsg 修正 268 269// String errMsg = "ファイル登録エラー!" 270 String errMsg = getResource().getLabel( "ERR0044" ) // 5.8.8.0 (2015/06/05) 271 + ex.getMessage() + HybsSystem.CR // 5.3.7.0 (2011/07/01) errMsg 修正 272 + "(" + toString() + HybsSystem.CR 273 + "FileURL=" + fileURL + ")"; 274 throw new HybsSystemException( errMsg,ex ); // 3.5.5.4 (2004/04/15) 引数の並び順変更 275 } 276 277 return(EVAL_PAGE); 278 } 279 280 /** 281 * タグリブオブジェクトをリリースします。 282 * キャッシュされて再利用されるので、フィールドの初期設定を行います。 283 * 284 * @og.rev 2.0.0.4 (2002/09/27) カスタムタグの release() メソッドを、追加 285 * @og.rev 3.0.1.1 (2003/03/06) columns を廃止 286 * @og.rev 3.1.1.2 (2003/04/04) Tomcat4.1 対応。release2() を doEndTag()で呼ぶ。 287 * @og.rev 3.5.4.2 (2003/12/15) ファイル名を指定できるようにします。 288 * @og.rev 5.6.5.2 (2013/06/21) columns 属性の追加 289 * @og.rev 5.6.5.2 (2013/06/21) useBackup 属性の追加 290 * 291 */ 292 @Override 293 protected void release2() { 294 super.release2(); 295 fileURL = HybsSystem.sys( "FILE_URL" ); 296 maxPostSize = 10*1024*1024; // 最大ファイル容量 10MB 297 tableId = HybsSystem.TBL_MDL_KEY ; 298 filename = null; // 3.5.4.2 (2003/12/15) 299 columns = null; // 5.6.5.2 (2013/06/21) 300 useBackup = false; // 5.6.5.3 (2013/06/28) 301 storageType = null; 302 bucketName = null; 303 } 304 305 /** 306 * ファイルアップロードの実行結果を DBTableModel に記述します。 307 * 308 * ここでは、"KEY","VALUE","ISFILE" のカラムに対して、値を設定していきます。 309 * 同時に、RequestAttribute 変数に、これらの値をセットすることで、 310 * {@XXXX} で値が取り出せる様にしています。 311 * 312 * @og.rev 2.2.0.0 (2002/12/17) 中国語(国際化)対応 エンコードの取得方法変更 313 * @og.rev 3.0.1.1 (2003/03/06) request 情報から{@XXXX} で値が取り出せる様に修正。 314 * @og.rev 3.5.2.0 (2003/10/20) カラム名(KEY,VALUE)に ISFILE を追加 315 * @og.rev 3.5.6.5 (2004/08/09) MultipartRequest 変更に伴なう修正(Enum変更、元ファイル名取得) 316 * @og.rev 3.5.6.6 (2004/08/23) 上記変更時のバグ修正。 317 * @og.rev 3.5.6.6 (2004/08/23) 元ファイルのキーを、XXXX_ORG にします。 318 * @og.rev 4.0.0.0 (2007/10/12) テーブルモデルの登録方法を変更 319 * @og.rev 5.3.2.0 (2011/02/01) チェック行のパラメーターはint配列側に変換して復元する。パラメーター名を保存する。 320 * @og.rev 5.4.4.2 (2012/02/03) CommonTagSupportと同様のチェックボックス判定を行う 321 * @og.rev 5.7.1.1 (2013/12/13) HTML5 ファイルアップロードの複数選択(multiple)対応 322 * @og.rev 5.7.1.2 (2013/12/20) 5.7.1.2 (2013/12/20) zip対応で、UploadedFile のメソッド変更 323 * @og.rev 5.7.3.0 (2014/02/07) zip対応の修正で、取得ファイル名が異なっていた。 324 * @og.rev 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 325 * @og.rev 5.9.25.0 (2017/10/06) クラウド対応 326 * 327 * @param multi MultipartRequestオブジェクト 328 * 329 * @return テーブルモデル 330 */ 331 private DBTableModel makeDBTable( final MultipartRequest multi ) { 332 333 DBTableModel table = DBTableModelUtil.newDBTable(); 334 335 table.init( names.length ); 336 337 for( int i=0; i<names.length; i++ ) { 338 DBColumn dbColumn = getDBColumn( names[i] ); 339 table.setDBColumn( i,dbColumn ); 340 } 341 342 String[] values ; // 4.0.0.0 (2007/10/12) 343 List<String> prmNames = new ArrayList<String>(); 344 345 // 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 346 StringBuilder buf = new StringBuilder(); 347 348 // 5.7.1.1 (2013/12/13) HTML5 ファイルアップロードの複数選択(multiple)対応 349// String[] files = multi.getFilenames(); // 3.6.0.0 (2004/09/17) 350 UploadedFile[] files = multi.getUploadedFile(); 351 for( int i=0; i<files.length; i++ ) { 352// String name = files[i]; 353// File fin = multi.getFile(name); 354// File fin = files[i].getFile(); 355// if( fin != null ) { 356// String val = multi.getFilesystemName(name); 357 String name = files[i].getName(); // multiple対応では、キーがかぶることがある。 358// String val = files[i].getFilesystemName(); 359// String val = files[i].getUploadFile().getPath(); // 5.7.1.2 (2013/12/20) zip対応で、UploadedFile のメソッド変更 360// String val = files[i].getUploadFile().getName(); // 5.7.3.0 (2014/02/07) 取得ファイル名が異なっていた。 361 String val = files[i].getUploadFile(); // 5.9.25.0 (2017/10/06) MODIFY getUploadFileで取得される値をFileからStringに変更対応 362 363 // 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 364 if( i==0 ) { buf.append( val ); } 365 else { buf.append( "," ).append( val ); } // カンマ結合で、最初だけ結合しない。 366 367 // "KEY","VALUE","ISFILE" の順にデータを作成します。 368 values = new String[] { name, val, "1" }; 369 table.addColumnValues( values ); 370 setRequestAttribute( name,val ); 371 prmNames.add( name ); // 5.7.1.1 (2013/12/13) List に設定する。 372 373 String orgName = name + "_ORG" ; 374// val = multi.getOriginalFileName(name); // 注意:取得は、送信名 375 val = files[i].getOriginalFileName(); // 注意:取得は、送信名 376 377 // "KEY","VALUE","ISFILE" の順にデータを作成します。 378 values = new String[] { orgName, val, "2" }; 379 table.addColumnValues( values ); 380 setRequestAttribute( orgName,val ); 381// } 382 } 383 384 // 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 385 setRequestAttribute( UPLOAD_FILES,buf.toString() ); 386 387 // "KEY","VALUE","ISFILE" の順にデータを作成します。 388 values = new String[] { "directory", fileURL, "0" }; 389 table.addColumnValues( values ); 390 setRequestAttribute( "directory",fileURL ); 391 392 String[] params = multi.getParameterNames(); 393 for( int i=0; i<params.length; i++ ) { 394 String name = params[i]; 395 // 5.3.2.0 (2011/02/01) チェック行のパラメーターはint配列側に変換 396 if ( HybsSystem.ROW_SEL_KEY.equals( name ) ) { 397 setRequestAttribute( name,multi.getIntParameters(name) ); 398 } 399 else { 400 // 5.6.5.2 (2013/06/21) チェックボックス配列の値取得を考慮した、MultipartRequest のパラメータ値取得 401 String val = getParamVal( name,multi ); 402// String val = multi.getParameter(name); 403// // "KEY","VALUE","ISFILE" の順にデータを作成します。 404// if( "0".equals(val) ){ // 5.4.4.2 チェックボックス配列対応。 405// String[] vals = multi.getParameters(name); 406// if( vals != null && vals.length > 1 ) { 407// for( int j=0; j<vals.length; j++ ) { 408// if( "1".equals( vals[j] ) ) { 409// val = "1"; 410// break; 411// } 412// } 413// } 414// } 415 416 values = new String[] { name, val, "0" }; 417 table.addColumnValues( values ); 418 setRequestAttribute( name,val ); 419 prmNames.add( name ); // 5.7.1.1 (2013/12/13) List に設定する。 420 } 421 } 422 423 // 5.3.2.0 (2011/02/01) パラメーター名を保存する。 424 // 5.7.1.1 (2013/12/13) List に設定する。 425// String[] names = new String[ files.length + params.length ]; 426// System.arraycopy( files, 0, names, 0, files.length ); 427// System.arraycopy( params, 0, names, files.length, params.length ); 428// setParameterNames( names ); 429 setParameterNames( prmNames.toArray( new String[prmNames.size()] ) ); 430 431 return table ; 432 } 433 434 /** 435 * ファイルアップロードの実行結果を 横持の DBTableModel に記述します。 436 * 437 * この処理は、columns 属性を設定した場合のみとします。 438 * 439 * DBTableModel作成時に、指定のカラムの"_01"〜"_99"の添え字をレコードとして作成します。 440 * 現状は、"KEY","VALUE","ISFILE" のカラムに、データを縦持ちで作成しています。 441 * これを、横持で作成しますが、カラムの末尾に、"_01"〜"_99" までの添え字を 442 * 持つ場合は、これをレコードと認識させます。 443 * 添え字がない場合は、カラムだけ作成されます。カラム名と同じリクエストがあれば、 444 * すべてのレコードに同じ値がセットされます。 445 * 446 * @og.rev 5.6.5.2 (2013/06/21) 新規作成 447 * @og.rev 5.6.6.1 (2013/07/12) 添え字がない場合の処理の見直し 448 * @og.rev 5.7.1.2 (2013/12/20) zip対応で、UploadedFile のメソッド変更 449 * @og.rev 5.7.3.0 (2014/02/07) zip対応の修正で、取得ファイル名が異なっていた。 450 * @og.rev 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 451 * @og.rev 5.9.25.0 (2017/10/06) クラウド対応 452 * 453 * @param multi MultipartRequestオブジェクト 454 * 455 * @return テーブルモデル 456 */ 457 private DBTableModel makeDBTableFromClms( final MultipartRequest multi ) { 458 459 DBTableModel table = DBTableModelUtil.newDBTable(); 460 461 String[] clmNames = columns.split( "," ); 462 463 table.init( clmNames.length ); 464 465 // 値配列(1行分) 466 String[] rowVal = new String[clmNames.length]; 467 468 // 5.7.1.1 (2013/12/13) HTML5 ファイルアップロードの複数選択(multiple)対応 469 UploadedFile[] files = multi.getUploadedFile(); 470 471 List<String> prmNames = new ArrayList<String>(); 472 473 for( int i=0; i<clmNames.length; i++ ) { 474 String clm = clmNames[i] ; 475 DBColumn dbColumn = getDBColumn( clm ); 476 table.setDBColumn( i,dbColumn ); 477 478 // 先に、カラム名と一致するパラメータを初期値としてセットしておきます。 479 String val = getParamVal( clm,multi ); 480 481 // 5.6.6.1 (2013/07/12) ファイル名も同様に、あれば初期値セットしておきます。 482 if( val == null ) { 483 // 5.7.1.1 (2013/12/13) HTML5 ファイルアップロードの複数選択(multiple)対応 484// val = multi.getFilesystemName( clm ); 485 for( int j=0; j<files.length; j++ ) { 486 String nm = files[j].getName(); 487 if( clm.equalsIgnoreCase( nm ) ) { 488// val = files[j].getFilesystemName(); 489// val = files[i].getUploadFile().getPath(); // 5.7.1.2 (2013/12/20) zip対応で、UploadedFile のメソッド変更 490 //val = files[i].getUploadFile().getName(); // 5.7.3.0 (2014/02/07) 取得ファイル名が異なっていた。 491 val = files[i].getUploadFile(); // 5.9.25.0 (2017/10/06) getUploadFileの型をFileからStringに変更対応 492 break; // 5.7.6.3 (2014/05/23) たぶん有ったほうが良い。 493 } 494 } 495 } 496 // 5.7.1.1 (2013/12/13) getFilesystemName() の中に、newFile が null の場合は、original を返す処理がある。 497// if( val == null ) { 498// // 5.7.1.1 (2013/12/13) HTML5 ファイルアップロードの複数選択(multiple)対応 499//// val = multi.getOriginalFileName( clm ); // 注意:取得は、送信名 500// for( int j=0; j<files.length; j++ ) { 501// String nm = files[j].getName(); 502// if( clm.equalsIgnoreCase( nm ) ) { 503// val = files[j].getOriginalFileName(); 504// break; 505// } 506// } 507// } 508 rowVal[i] = ( val == null ) ? "" : val ; 509 } 510 511 // 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 512 StringBuilder buf = new StringBuilder(); 513 514// String[] files = multi.getFilenames(); // 3.6.0.0 (2004/09/17) 515 for( int i=0; i<files.length; i++ ) { 516// String name = files[i]; 517// File fin = multi.getFile(name); 518 String name = files[i].getName(); 519// File fin = files[i].getFile(); 520// if( fin != null ) { 521 // 5.6.6.1 (2013/07/12) 添え字がない場合の処理の見直し。先にレコードを作成 522 String[] values = new String[clmNames.length]; 523 System.arraycopy( rowVal,0,values,0,values.length ); // 行にセットするに当たり、rowVal を values にコピーしておく。 524 525 // ファイル名を Attribute で使えるようにセットしておく。 526// String fval = multi.getFilesystemName(name); 527// String fval = files[i].getFilesystemName(); 528// String fval = files[i].getUploadFile().getPath(); // 5.7.1.2 (2013/12/20) zip対応で、UploadedFile のメソッド変更 529// String fval = files[i].getUploadFile().getName(); // 5.7.3.0 (2014/02/07) 取得ファイル名が異なっていた。 530 String fval = files[i].getUploadFile(); // 5.9.25.0 (2017/10/06) getUploadで取得される値をFileからStringに変更 531 532 setRequestAttribute( name,fval ); 533 prmNames.add( name ); // 5.7.1.1 (2013/12/13) List に設定する。 534 535 // 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 536 if( i==0 ) { buf.append( fval ); } 537 else { buf.append( "," ).append( fval ); } // カンマ結合で、最初だけ結合しない。 538 539 String orgName = name + "_ORG" ; 540// String oval = multi.getOriginalFileName(name); // 注意:取得は、送信名 541 String oval = files[i].getOriginalFileName(); // 注意:取得は、送信名 542 setRequestAttribute( orgName,oval ); 543 544 // ファイルのキーを元に、添え字を検索します。 545 int adrs = name.lastIndexOf( '_' ); // 添え字は、'_' で区切られます。 546 // 5.6.6.1 (2013/07/12) 添え字がない場合の処理の見直し。後続処理を行う。 547// if( adrs < 0 ) { continue; } // 添え字がない 548 if( adrs > 0 ) { 549 String fnm = name.substring( 0,adrs ); // ファイル名(分割後) 550 String sub = name.substring( adrs ); // 添え字(アンダーバー含む) 551 552// String[] values = new String[clmNames.length]; 553// System.arraycopy( rowVal,0,values,0,values.length ); // 行にセットするに当たり、rowVal を values にコピーしておく。 554 555 // カラム名で検索しながら、レコード単位になるようにセットしていきます。 556 for( int j=0; j<clmNames.length; j++ ) { 557 String clm = clmNames[j] ; 558 String nm = null; 559 String val = null; 560 561 if( fnm.equalsIgnoreCase( clm ) ) { // ファイル名カラム(_NEWファイル名も、この値にすでに変わっている) 562// nm = name; 563// val = multi.getFilesystemName( nm ); 564 val = fval; 565 } 566 else if( ( fnm + "_ORG" ).equalsIgnoreCase( clm ) ) { // 元ファイル名カラム 567// nm = name + "_ORG" ; 568// val = multi.getOriginalFileName( name ); // 注意:取得は、送信名 569 val = oval; 570 } 571 else if( ( fnm + "_NEW" ).equalsIgnoreCase( clm ) ) { // 新ファイル名カラム 572 nm = name + "_NEW" ; 573 val = multi.getParameter( nm ); 574 } 575 else { 576 nm = clmNames[j] + sub; // sub は、アンダーバー含む添え字 577 // 5.6.5.2 (2013/06/21) チェックボックス配列の値取得を考慮した、MultipartRequest のパラメータ値取得 578 val = getParamVal( nm,multi ); 579 } 580 if( val != null ) { values[j] = val ; } // val が null でないときのみセットします。 581// setRequestAttribute( nm,val ); 582 } 583 } 584 table.addColumnValues( values ); 585// } 586 } 587 // 5.7.6.3 (2014/05/23) アップロードファイルのCSVセット 588 setRequestAttribute( UPLOAD_FILES,buf.toString() ); 589 590 // Attribute で使えるようにセットしておく。 591 setRequestAttribute( "directory",fileURL ); 592 593 // Attribute で使えるようにセットしておく。 594 String[] params = multi.getParameterNames(); 595 for( int i=0; i<params.length; i++ ) { 596 String name = params[i]; 597 // 5.3.2.0 (2011/02/01) チェック行のパラメーターはint配列側に変換 598 if ( HybsSystem.ROW_SEL_KEY.equals( name ) ) { 599 setRequestAttribute( name,multi.getIntParameters(name) ); 600 } 601 else { 602 // 5.6.5.2 (2013/06/21) チェックボックス配列の値取得を考慮した、MultipartRequest のパラメータ値取得 603 String val = getParamVal( name,multi ); 604 setRequestAttribute( name,val ); 605 prmNames.add( name ); // 5.7.1.1 (2013/12/13) List に設定する。 606 } 607 } 608 609 // 5.3.2.0 (2011/02/01) パラメーター名を保存する。 610 // 5.7.1.1 (2013/12/13) List に設定する。 611// String[] names = new String[ files.length + params.length ]; 612// System.arraycopy( files, 0, names, 0, files.length ); 613// System.arraycopy( params, 0, names, files.length, params.length ); 614// setParameterNames( names ); 615 setParameterNames( prmNames.toArray( new String[prmNames.size()] ) ); 616 617 return table ; 618 } 619 620 /** 621 * チェックボックス配列の値取得を考慮した、MultipartRequest のパラメータ値取得 622 * 623 * ここでは、、MultipartRequest のパラメータ値を取得します。 624 * 値の取得で、チェックボックス配列の場合は、取得した値が、"0" の場合のみ 625 * 配列でパラメータを取得し直し、"1" がないかどうか再度検索します。 626 * チェックボックスでは、チェック時の "1" と、チェックされなかった場合の、 627 * hidden の "0" の両方の値が配列としてリクエストされるケースがある為です。 628 * 629 * @og.rev 5.6.5.2 (2013/06/21) 新規作成 630 * 631 * @param key MultipartRequest のパラメータ取得のためのキー 632 * @param multi MultipartRequestオブジェクト 633 * 634 * @return チェックボックス配列を考慮したパラメータ値 635 */ 636 private String getParamVal( final String key , final MultipartRequest multi ) { 637 String val = multi.getParameter( key ); 638 639 if( "0".equals(val) ){ // 5.4.4.2 チェックボックス配列対応。 640 String[] vals = multi.getParameters( key ); 641 if( vals != null && vals.length > 1 ) { 642 for( int j=0; j<vals.length; j++ ) { 643 if( "1".equals( vals[j] ) ) { 644 val = "1"; 645 break; 646 } 647 } 648 } 649 } 650 return val ; 651 } 652 653 /** 654 * 【TAG】ファイルをアップロードするディレクトリを指定します 655 * (初期値:FILE_URL[={@og.value org.opengion.hayabusa.common.SystemData#FILE_URL}])。 656 * 657 * @og.tag 658 * この属性で指定されるディレクトリに、アップロードされたファイルをセーブします。 659 * 指定方法は、通常の fileURL 属性と同様に、先頭が、'/' (UNIX) または、2文字目が、 660 * ":" (Windows)の場合は、指定のURLそのままのディレクトリに、そうでない場合は、 661 * fileURL = "{@USER.ID}" と指定すると、FILE_URL 属性で指定のフォルダの下に、 662 * さらに、各個人ID別のフォルダを作成して、そこにセーブします。 663 * (初期値:システム定数のFILE_URL[={@og.value org.opengion.hayabusa.common.SystemData#FILE_URL}])。 664 * 665 * @og.rev 4.0.0.0 (2005/01/31) StringUtil.urlAppend メソッドの利用 666 * @og.rev 4.0.0.0 (2007/11/20) 指定されたディレクトリ名の最後が"\"or"/"で終わっていない場合に、"/"を付加する。 667 * @og.rev 5.7.1.1 (2013/12/13) リクエスト変数が使えないエラーを表示する。 668 * @og.rev 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェックを行う。 669 * @og.rev 5.9.13.1 (2016/10/21) チェックは行わない 670 * 671 * @param url ファイルURL 672 * @see org.opengion.hayabusa.common.SystemData#FILE_URL 673 */ 674 public void setFileURL( final String url ) { 675 String furl = nval( getRequestParameter( url ),null ); 676 if( furl != null ) { 677 char ch = furl.charAt( furl.length()-1 ); 678 if( ch != '/' && ch != '\\' ) { furl = furl + "/"; } 679 fileURL = StringUtil.urlAppend( fileURL,furl ); 680 } 681// else if( url != null && url.startsWith( "{@" ) ) { 682// String errMsg = "upload では、enctype=\"multipart/form-data\" のため、{@XXXX}形式のパラメータが使えません。" 683// + " fileURL=[" + url + "]" ; 684// throw new HybsSystemException( errMsg ); // 5.7.1.1 (2013/12/13) リクエスト変数が使えないエラー 685// } 686 else { 687// chckReqParam( url,"fileURL" ); // 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェック 688 // 5.9.13.1 (2016/10/21) このチェックがあると既存システムのバージョンアップ時に気が付かずに 689 // 対象画面以外でエラーを発生させてしまう可能性があるので、V5では消しておく 690 } 691 } 692 693 /** 694 * 【TAG】最大転送サイズ(Byte)を指定します(初期値:10485760)。 695 * 696 * @og.tag 697 * 最大転送サイズを指定します。初期値は、10*1024*1024 = 10MB です。 698 * 指定は、Byte 単位で指定します。 699 * 0,またはマイナスを指定することで、制限チェックを外す(=無制限)事ができます。 700 * 701 * @og.rev 3.0.1.1 (2003/03/06) maxPostSize の設定バグ修正。 702 * @og.rev 5.6.5.3 (2013/06/28) コメント追加(0,またはマイナスで無制限) 703 * @og.rev 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェックを行う。 704 * 705 * @param maxPS 最大転送サイズ 706 */ 707 public void setMaxPostSize( final String maxPS ) { 708 maxPostSize = nval( getRequestParameter( maxPS ),maxPostSize ); 709 710 chckReqParam( maxPS,"maxPostSize" ); // 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェック 711 } 712 713 /** 714 * 【TAG】(通常は使いません)結果のDBTableModelを、sessionに登録するときのキーを指定します 715 * (初期値:HybsSystem#TBL_MDL_KEY[={@og.value org.opengion.hayabusa.common.HybsSystem#TBL_MDL_KEY}])。 716 * 717 * @og.tag 718 * 検索結果より、DBTableModelオブジェクトを作成します。これを、下流のviewタグ等に 719 * 渡す場合に、通常は、session を利用します。その場合の登録キーです。 720 * query タグを同時に実行して、結果を求める場合、同一メモリに配置される為、 721 * この tableId 属性を利用して、メモリ空間を分けます。 722 * (初期値:HybsSystem#TBL_MDL_KEY[={@og.value org.opengion.hayabusa.common.HybsSystem#TBL_MDL_KEY}])。 723 * 724 * @og.rev 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェックを行う。 725 * 726 * @param id sessionに登録する時の ID 727 */ 728 public void setTableId( final String id ) { 729 tableId = nval( getRequestParameter( id ),tableId ); 730 731 chckReqParam( id,"tableId" ); // 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェック 732 } 733 734 /** 735 * 【TAG】(通常使いません)ファイルを作成するときのファイル名をセットします。 736 * 737 * @og.tag 738 * ファイルを作成するときのファイル名をセットします。 739 * これは、複数同時にアップロードファイル名を変更する時に使用できません。 740 * 通常、アップロードされたファイル名を指定する場合、アップロードするinput タグの 741 * name 属性に指定する名称 + "_NEW" というリクエスト値を同時に送信すれば、 742 * 内部的に関連付けて、ファイル名を更新します。 743 * その場合、クライアントより指定したファイル名は、name属性+"_ORG" という 744 * リクエスト値として取得することが可能になります。 745 * name属性 には、最終的に設定されたファイル名がセットされています。 746 * いずれの値も、{@name属性+"_ORG"} や、{@name属性+"_NEW"}として、 747 * アップロードのオリジナルと変更後のファイル名を取得することが出来ます。 748 * 749 * 5.7.1.2 (2013/12/20) zip 対応 750 * filename 属性に、".zip" の拡張子のファイル名を指定した場合は、アップロードされた一連のファイルを 751 * ZIP圧縮します。これは、アップロード後の処理になります。 752 * ZIP圧縮のオリジナルファイルは、そのまま残ります。 753 * なお、ZIPファイルは、useBackup属性を true に設定しても、無関係に、上書きされます。 754 * 755 * 5.7.4.3 (2014/03/28) filename 属性のリクエスト変数対応 756 * filename 属性のみ、{@XXXX} のリクエスト変数が使えるようにします。 757 * 他のパラメータでは使えません。 758 * これは、multipart/form-data のリクエストでは、パートの分解処理をしないと、リクエスト変数が 759 * 拾えない為、リクエスト変数は、この、upload タグ以降でのみ利用可能でした。 760 * zip対応と関連付けて、filename 属性のみ、利用できるように、MultipartRequest 側の処理に組み込みます。 761 * 762 * @og.rev 3.5.4.2 (2003/12/15) ファイル名を指定できるようにします。 763 * @og.rev 5.7.1.1 (2013/12/13) リクエスト変数が使えないエラーを表示する。 764 * @og.rev 5.7.4.3 (2014/03/28) リクエスト変数を使えるようにします。 765 * 766 * @param fname ファイル名 767 */ 768 public void setFilename( final String fname ) { 769 filename = nval( getReservedParameter( fname ),null ); // 予約語のみ処理をします。 770 771// filename = nval( getRequestParameter( fname ),null ); 772 773// if( filename == null && fname != null && fname.startsWith( "{@" ) ) { 774// String errMsg = "upload では、enctype=\"multipart/form-data\" のため、{@XXXX}形式のパラメータが使えません。" 775// + " filename=[" + fname + "]" ; 776// throw new HybsSystemException( errMsg ); // 5.7.1.1 (2013/12/13) リクエスト変数が使えないエラー 777// } 778 } 779 780 /** 781 * 【TAG】DBTableModel作成時に、指定のカラムの"_01"〜"_99"の添え字をレコードとして作成します。 782 * 783 * @og.tag 784 * 現状は、"KEY","VALUE","ISFILE" のカラムに、データを縦持ちで作成しています。 785 * これを、横持で作成しますが、カラムの末尾に、"_01"〜"_99" までの添え字を 786 * 持つ場合は、これをレコードと認識させます。 787 * アンダーバーがない場合は、カラムだけ作成されます。カラム名と同じリクエストがあれば、 788 * すべてのレコードに同じ値がセットされます。 789 * この処理は、columns 属性を設定した場合のみとします。 790 * 791 * @og.rev 5.6.5.2 (2013/06/21) 新規作成 792 * @og.rev 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェックを行う。 793 * 794 * @param clms DBTableModel作成時のカラム列(カンマ区切り文字) 795 */ 796 public void setColumns( final String clms ) { 797 columns = nval( getRequestParameter( clms ),columns ); 798 799 chckReqParam( clms,"columns" ); // 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェック 800 } 801 802 /** 803 * 【TAG】ファイルアップロード時に、すでに同名のファイルが存在した場合に、バックアップ処理(renameTo)するかどうか[true/false]を指定します(初期値:false)。 804 * 805 * @og.tag 806 * ファイルアップロード時に、アップロード先に、同名のファイルが存在した場合は、既存機能は、そのまま 807 * 置き換えていましたが、簡易バージョンアップ機能として、useBackup="true" を指定すると、既存のファイルを 808 * リネームして、バックアップファイルを作成します。 809 * バックアップファイルは、アップロードフォルダを基準として、_backup/ファイル名.拡張子_処理時刻のlong値.拡張子 になります。 810 * オリジナルのファイル名(拡張子付)を残したまま、"_処理時刻のlong値" を追加し、さらに、オリジナルの拡張子を追加します。 811 * バックアップファイルの形式は指定できません。 812 * 813 * 初期値は、互換性を持たせるため、false です。 814 * 815 * @og.rev 5.6.5.3 (2013/06/28) 新規作成 816 * @og.rev 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェックを行う。 817 * 818 * @param flag ファイルアップロード時に、バックアップ処理(renameTo)するかどうか[true/false]を指定 819 */ 820 public void setUseBackup( final String flag ) { 821 useBackup = nval( getRequestParameter( flag ),useBackup ); 822 823 chckReqParam( flag,"useBackup" ); // 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェック 824 } 825 826 /** 827 * リクエスト変数が使えない事のチェックを行います。 828 * 829 * upload では、enctype="multipart/form-data" のため、{@XXXX}形式のパラメータが使えません。 830 * 5.7.4.3 (2014/03/28) から、filename のみ利用可能としたことで、同様に利用できると 831 * 勘違いするケースに対応する為、すべてのパラメータについてチェックを行います。 832 * ここでは、getRequestParameter( String ) の実行後、すぐに、isNull() 判定を行う事で、 833 * リクエスト変数の存在チェックを行う事にしています。 834 * 835 * @og.rev 5.7.4.3 (2014/03/28) リクエスト変数が使えない事のチェックを行う。 836 * 837 * @param org 引数のオリジナル値 838 * @param key エラーの発生した変数名 839 * @throws HybsSystemException パラメータが使用されていた場合 840 */ 841 private void chckReqParam( final String org,final String key ) { 842 if( isNull() && org != null && org.contains( "{@" ) ) { 843 String errMsg = "upload では、enctype=\"multipart/form-data\" のため、{@XXXX}形式のパラメータが使えません。" 844 + HybsSystem.CR 845 + " " + key + "=[" + org + "]" ; 846 throw new HybsSystemException( errMsg ); // リクエスト変数が使えないエラー 847 } 848 } 849 850 /** 851 * タグの名称を、返します。 852 * 自分自身のクラス名より、自動的に取り出せないため、このメソッドをオーバーライドします。 853 * 854 * @og.rev 4.0.0.0 (2005/01/31) 新規追加 855 * 856 * @return タグの名称 857 */ 858 @Override 859 protected String getTagName() { 860 return "upload" ; 861 } 862 863 /** 864 * 【TAG】保存先ストレージタイプを設定します。 865 * 866 * @og.tag 867 * ファイルを読み取り元の、ストレージタイプを設定します。 868 * 未設定の場合は、システムリソースの「CLOUD_TARGET」が参照されます。 869 * 自身のサーバを指定する場合は、「default」を設定してください。 870 * 871 * @og.rev 5.10.9.0 (2019/03/01) 新規追加 872 * 873 * @param storage 保存先ストレージタイプ 874 */ 875 public void setStorageType( final String storage ) { 876 storageType = nval( getRequestParameter(storage), storageType ); 877 } 878 879 /** 880 * 【TAG】保存先バケット名を設定します。 881 * 882 * @og.tag 883 * ファイルを読み取り元の、バケット名を指定します。 884 * クラウドストレージ利用時のみ有効です。 885 * 未設定の場合は、システムリソースの「CLOUD_BUKET」が参照されます。 886 * 887 * @og.rev 5.10.9.0 (2019/03/01) 新規追加 888 * 889 * @param bucket 保存先バケット名 890 */ 891 public void setBucketName( final String bucket ) { 892 bucketName = nval( getRequestParameter(bucket), bucketName ); 893 } 894 895 /** 896 * このオブジェクトの文字列表現を返します。 897 * 基本的にデバッグ目的に使用します。 898 * 899 * @og.rev 5.10.9.0 (2019/03/01) 出力対象に、storageType, bucketNameを追加。 900 * 901 * @return このクラスの文字列表現 902 */ 903 @Override 904 public String toString() { 905 return org.opengion.fukurou.util.ToString.title( this.getClass().getName() ) 906 .println( "VERSION" ,VERSION ) 907 .println( "names" ,names ) 908 .println( "ENCODE" ,ENCODE ) 909 .println( "fileURL" ,fileURL ) 910 .println( "filename" ,filename ) 911 .println( "maxPostSize" ,maxPostSize) 912 .println( "tableId" ,tableId ) 913 .println( "columns" ,columns ) // 5.6.5.2 (2013/06/21) 914 .println( "useBackup" ,useBackup ) // 5.6.5.3 (2013/06/28) 915 .println( "storageType" ,storageType) 916 .println( "bucketName" ,bucketName ) 917 .println( "Other..." ,getAttributes().getAttribute() ) 918 .fixForm().toString() ; 919 } 920}