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.db;
017
018import java.sql.DatabaseMetaData;
019import java.sql.SQLException;
020import java.util.Locale;
021import java.util.Properties;
022import static org.opengion.fukurou.system.HybsConst.CR;                         // 6.1.0.0 (2014/12/26) refactoring
023import static org.opengion.fukurou.system.HybsConst.BUFFER_MIDDLE;      // 6.1.0.0 (2014/12/26) refactoring
024import org.opengion.fukurou.system.OgRuntimeException ;         // 6.4.2.0 (2016/01/29)
025
026/**
027 * JAXBで自動生成されたDBIDクラスを拡張したクラスです。
028 * (継承しているわけではない)
029 * <BR>
030 * 以下の属性が追加されています。<BR>
031 * dbProductName<BR>
032 * dbProductVersion<BR>
033 * driverName<BR>
034 * driverVersion<BR>
035 *
036 * @og.rev 4.0.0.0 (2007/10/25) 新規作成
037 * @og.rev 5.1.7.0 (2010/06/01) メソッドの修正、Cloneable の追加(浅いコピー)
038 * @og.rev 5.6.6.0 (2013/07/05) 表題(title)属性を追加
039 *
040 * @version  4.0
041 * @author       高橋正和
042 * @since        JDK5.0,
043 */
044public class EDbid implements Cloneable {
045
046        private         String  dbidKey                 = "DEFAULT";
047        private         String  title                   ;                                               // 5.6.6.0 (2013/07/05) 表題(title)属性を追加
048        private         String  url                             ;
049        private         String  user                    ;
050        private         String  password                = "";                                   // 5.7.2.0 (2014/01/10) パスワードは空文字を初期値とする。
051        private         boolean readonly                ;
052        private         int             mincount                = 3;
053        private         int             maxcount                = 30;
054        private         int             pooltime                = 3000;         // (秒数)
055        private         boolean isUseAppInfo    = true;
056        private         boolean isParamMetaData ;                                               // 5.3.8.0 (2011/08/01)
057
058        private         String  dbProductName   ;
059        private         String  dbProductVersion;
060        private         String  driverName              ;
061        private         String  driverVersion   ;
062
063        /* 5.5.2.0 (2012/05/01) DBConfigでpropertiesの指定を可能にします */
064        private final Properties props = new Properties();
065
066        /**
067         * デフォルトコンストラクター
068         *
069         * @og.rev 6.4.2.0 (2016/01/29) PMD refactoring. Each class should declare at least one constructor.
070         */
071        public EDbid() { super(); }             // これも、自動的に呼ばれるが、空のメソッドを作成すると警告されるので、明示的にしておきます。
072
073        /**
074         * dbidKeyの取得
075         *
076         * 内部的には、大文字のみで管理します。
077         *
078         * @return      dbidキー
079         */
080        public String getDbidKey() {
081                return dbidKey;
082        }
083
084        /**
085         * dbidキーの設定
086         *
087         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
088         * 内部的には、大文字のみで管理します。
089         *
090         * @og.rev 4.1.0.1 (2008/01/21) 登録時に、大文字に変換する。
091         *
092         * @param value 接続先ID
093         */
094        protected void setDbidKey( final String value ) {
095                if( isNotNull( value ) ) { dbidKey = value.toUpperCase( Locale.JAPAN ); }
096        }
097
098        /**
099         * 表題(title)属性の取得
100         *
101         * この、dbidKey を表す表題を取得します。ラベル(名前)のようなものです。
102         *
103         * @og.rev 5.6.6.0 (2013/07/05) 新規追加
104         *
105         * @return      表題(title)
106         */
107        public String getTitle() {
108                return title;
109        }
110
111        /**
112         * 表題(title)の設定
113         *
114         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
115         *
116         * @og.rev 5.6.6.0 (2013/07/05) 新規追加
117         * @og.rev 5.6.8.0 (2013/09/06) title が未設定の時は、dbidKey をセットしておきます。
118         *
119         * @param value 表題(title)
120         */
121        protected void setTitle( final String value ) {
122                if( isNotNull( value ) ) { title = value; }
123                else                                     { title = dbidKey; }           // 5.6.8.0 (2013/09/06)
124        }
125
126        /**
127         * URLの取得。
128         *
129         * @return      URL
130         */
131        public String getUrl() {
132                return url;
133        }
134
135        /**
136         * URLの設定
137         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
138         *
139         * @param value URL文字列
140         */
141        protected void setUrl( final String value ) {
142                if( isNotNull( value ) ) { url = value; }
143        }
144
145        /**
146         * ユーザーの取得。
147         *
148         * @return      ユーザー
149         */
150        public String getUser() {
151                return user;
152        }
153
154        /**
155         * userの設定
156         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
157         *
158         * @param value user文字列
159         */
160        protected void setUser( final String value ) {
161                if( isNotNull( value ) ) { user = value; }
162        }
163
164        /**
165         * パスワードの取得。
166         *
167         * @return      パスワード
168         */
169        public String getPassword() {
170                return password;
171        }
172
173        /**
174         * パスワードの設定
175         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
176         *
177         * @param value パスワード文字列
178         */
179        protected void setPassword( final String value ) {
180                if( isNotNull( value ) ) { password = value; }
181        }
182
183        /**
184         * readonlyの取得。
185         *
186         * @return      [true:読み取り専用/false:通常]
187         */
188        public boolean isReadonly() {
189                return readonly ;
190        }
191
192        /**
193         * readonlyの設定
194         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
195         *
196         * @param value  readonly文字列 [true/false]
197         */
198        protected void setReadonly( final String value ) {
199                if( isNotNull( value ) ) { readonly = Boolean.parseBoolean( value ); }          // 6.1.0.0 (2014/12/26) refactoring
200        }
201
202        /**
203         * 最小数の取得。
204         *
205         * @return      最小数
206         */
207        public int getMincount() {
208                return mincount;
209        }
210
211        /**
212         * 最小数の設定
213         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
214         *
215         * @param value 最小数(数字タイプ文字列)
216         */
217        protected void setMincount( final String value ) {
218                if( isNotNull( value ) ) { mincount = Integer.parseInt( value ); }              // 6.0.2.4 (2014/10/17) メソッド間違い
219        }
220
221        /**
222         * 最大数の取得。
223         *
224         * @return      最大数
225         */
226        public int getMaxcount() {
227                return maxcount;
228        }
229
230        /**
231         * 最大数の設定
232         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
233         *
234         * @param value 最大数(数字タイプ文字列)
235         */
236        protected void setMaxcount( final String value ) {
237                if( isNotNull( value ) ) { maxcount = Integer.parseInt( value ); }              // 6.0.2.4 (2014/10/17) メソッド間違い
238        }
239
240        /**
241         * プールに保管して置く時間(秒数)の取得。
242         *
243         * @return      プールに保管する時間(秒数)
244         */
245        public int getPooltime() {
246                return pooltime;
247        }
248
249        /**
250         * プールに保管して置く時間(秒数))の設定
251         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
252         *
253         * @param value プールに保管して置く時間(秒数)(数字タイプ文字列)
254         */
255        protected void setPooltime( final String value ) {
256                if( isNotNull( value ) ) { pooltime = Integer.parseInt( value ); }              // 6.0.2.4 (2014/10/17) メソッド間違い
257        }
258
259        /**
260         * ApplicationInfoオブジェクトを使用するかどうか[true/false]の情報を取得。
261         *
262         * @return      使用するかどうか
263         */
264        public boolean isApplicationInfo() {
265                return isUseAppInfo ;
266        }
267
268        /**
269         * ApplicationInfoオブジェクトを使用するかどうか[true/false]の情報を設定
270         * 引数が、null でなく、ゼロ文字列でもない場合のみ、セットします。
271         *
272         * @param value ApplicationInfoオブジェクトを使用するかどうか [true/false]
273         */
274        protected void setApplicationInfo( final String value ) {
275                if( isNotNull( value ) ) { isUseAppInfo = Boolean.parseBoolean( value ); }      // 6.1.0.0 (2014/12/26) refactoring
276        }
277
278        /**
279         * メタデータを受け取って dbProductName,dbProductVersion,driverName,driverVersion を一括で設定します。
280         *
281         * @og.rev 4.0.0.0 (2007/10/30) 保持情報オブジェクト化に伴う変更
282         * @og.rev 5.3.8.0 (2011/08/01) postgreSQL時のカラムタイプ判定の必要有無(ApplicationInfo.useParameterMetaData)
283         * @og.rev 6.4.9.5 (2016/09/09) DERBY も、isParamMetaData=true を返します。
284         *
285         * @param meta メタデータオブジェクト
286         * @throws SQLException データベースアクセスエラー
287         */
288        public void setMetaDataInfo( final DatabaseMetaData meta ) throws SQLException {
289                dbProductName   = meta.getDatabaseProductName() ;
290                dbProductVersion= meta.getDatabaseProductVersion() ;
291                driverName              = meta.getDriverName() ;
292                driverVersion   = meta.getDriverVersion() ;
293
294                // dbProductName は、大文字にしておいても問題は無いが、今は影響範囲を小さくしておきます。
295                final String dbName = dbProductName.toUpperCase( Locale.JAPAN );                // 6.4.9.5 (2016/09/09) 大文字化します。
296
297                isUseAppInfo    = isUseAppInfo && "ORACLE".equalsIgnoreCase( dbProductName ) ;
298                isParamMetaData = dbName.contains( "POSTGRES" )
299                                           || dbName.contains( "DERBY" );                                       // 6.4.9.5 (2016/09/09) 
300        }
301
302        /**
303         * DBプロダクト名の取得。
304         *
305         * @return      DBプロダクト名
306         */
307        public String getDbProductName() {
308                return dbProductName;
309        }
310
311        /**
312         * DBプロダクト・バージョンの取得。
313         *
314         * @return      DBプロダクト・バージョン
315         */
316        public String getDbProductVersion() {
317                return dbProductVersion;
318        }
319
320        /**
321         * ドライバー名の取得。
322         *
323         * @return      ドライバー名
324         */
325        public String getDriverName() {
326                return driverName;
327        }
328
329        /**
330         * ドライバーのバージョンの取得。
331         *
332         * @return      ドライバーのバージョン
333         */
334        public String getDriverVersion() {
335                return driverVersion;
336        }
337
338        /**
339         * ParamMetaData を利用するかどうか[true/false]を返します。
340         *
341         * これは、PostgreSQL の時は、trueになります。
342         *
343         * @og.rev 5.3.8.0 (2011/08/01) 新規追加
344         *
345         * @return      利用するかどうか[true/false]
346         */
347        public boolean useParamMetaData() {
348                return isParamMetaData ;
349        }
350
351        /**
352         * propMapへの追加。
353         *
354         * @og.rev 5.5.2.0 (2012/05/01) 新規追加
355         * @param key プロパティのキー
356         * @param val ポロパティの値
357         */
358        protected void addProp( final String key, final String val ) {
359                props.put ( key, val );
360        }
361
362        /**
363         * propMapへの追加。
364         * 最初の=でkeyとvalueを分割する
365         *
366         * @og.rev 5.5.2.0 (2012/05/01) 新規追加
367         * @og.rev 5.5.2.1 (2012/05/07) propsへの追加漏れ,カンマを=に変更
368         * @param prop プロパティのキーと値の組み合わせ
369         */
370        protected void addProp( final String prop ) {
371                if( prop!=null && prop.indexOf( '=' ) > 0 ){
372                        final String key = prop.substring( 0 , prop.indexOf('=') );
373                        final String val = prop.substring( prop.indexOf('=') + 1 );
374                        addProp( key, val ); // 5.5.2.1 (2012/05/07)
375                }
376        }
377
378        /**
379         * propMapの取得。
380         *
381         * @og.rev 5.5.2.0 (2012/05/01) 新規追加
382         * @og.rev 6.3.9.0 (2015/11/06) 内部Propertiesオブジェクトではなく、複製して返します。
383         *
384         * @return 複製Propertiesオブジェクト
385         */
386        public Properties getProps(){
387                return new Properties( props );                 // 6.3.9.0 (2015/11/06)
388        }
389
390        /**
391         * 文字列がnullでも空文字列でもないかどうかを評価します。
392         *
393         * スペース文字は、trim() してから、!isEmpty() で判断しますので、false になります。
394         *
395         * @og.rev 5.1.7.0 (2010/06/01) 新規作成
396         *
397         * @param value 評価する文字列
398         *
399         * @return 結果(true:nullでも空文字列でもない)
400         */
401        private boolean isNotNull( final String value ) {
402                return value != null && !value.trim().isEmpty() ;                       // 6.0.2.5 (2014/10/31) refactoring
403        }
404
405        /**
406         * 自分自身の clone を返します。
407         *
408         * Cloneable の実装で、内部的には、Object クラスの clone メソッドを読んでいます。
409         *
410         * @og.rev 5.1.7.0 (2010/06/01) 新規作成
411         *
412         * @return      自分自身の clone を返します。
413         * @throws      RuntimeException clone が失敗しました場合
414         * @og.rtnNotNull
415         */
416        @Override
417        public EDbid clone() {
418                try {
419                        return (EDbid)super.clone() ;
420                }
421                catch( final CloneNotSupportedException ex ) {
422                        final String errMsg = "clone が失敗しました。"
423                                                                + ex.getMessage() ;
424                        throw new OgRuntimeException( errMsg,ex );
425                }
426        }
427
428        /**
429         * 自分自身の clone を返します。
430         *
431         * ここでは、dbidKey を指定した、clone() を作成します。
432         * Titleも、指定した、dbidKey をセットしておきます。
433         *
434         * @og.rev 6.4.3.4 (2016/03/11) 新規作成
435         *
436         * @param id 接続先ID
437         * @return      dbidKeyを指定して、自分自身のcloneを返します。
438         * @og.rtnNotNull
439         */
440        public EDbid clone( final String id ) {
441                final EDbid obj = this.clone();
442                obj.setDbidKey( id );
443                obj.setTitle( id );
444
445                return obj;
446        }
447
448        /**
449         * 簡易な内部情報の文字列化します。
450         *
451         * @og.rev 5.3.4.0 (2011/04/01) toString() の簡易版
452         * @og.rev 5.5.2.1 (2012/05/07) properties情報を出力するようにする
453         * @og.rev 5.6.6.0 (2013/07/05) 表題(title)属性を追加
454         *
455         * @return 内部情報の文字列
456         * @og.rtnNotNull
457         */
458        public String info() {
459                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
460                        .append( "DBID=["  ).append( dbidKey ).append( "] , " )
461                        .append( "TITLE=[" ).append( title   ).append( "] , " )                 // 5.6.6.0 (2013/07/05) 表題(title)属性を追加
462                        .append( "URL=["   ).append( url     ).append( "] , " )
463                        .append( "USER=["  ).append( user    ).append( ']'  )
464                        .append( CR )
465                        .append( "DB Product=[" ).append( dbProductName )
466                        .append( '(' ).append( dbProductVersion ).append( ')' ).append( ']' )
467                        .append( CR )
468                        .append( "DB Driver =[" ).append( driverName )
469                        .append( '(' ).append( driverVersion ).append( ')' ).append( ']' )
470                        .append( CR )
471                        .append( "PROPERTIES=[" ).append( props.toString() ).append( ']' ) // 5.5.2.1 (2012/05/07)
472                        .append( CR );
473                return buf.toString();
474        }
475
476        /**
477         * 内部情報を文字列化します。
478         *
479         * @og.rev 5.3.4.0 (2011/04/01) info() メソッドを利用するように変更
480         *
481         * @return      内部情報の文字列
482         * @og.rtnNotNull
483         */
484        @Override
485        public String toString() {
486                final StringBuilder buf = new StringBuilder( BUFFER_MIDDLE )
487                        .append( info() )
488                //      .append( "DBID=[" ).append( dbidKey ).append( "]," )
489                //      .append( "URL =[" ).append( url     ).append( "]," )
490                //      .append( "USER=[" ).append( user    ).append( "]," )
491                //      .append( CR )
492                //      .append( "DB Product=[" ).append( dbProductName )
493                //      .append( "(" ).append( dbProductVersion ).append( ")" ).append( "]" )
494                //      .append( CR )
495                //      .append( "DB Driver =[" ).append( driverName )
496                //      .append( "(" ).append( driverVersion ).append( ")" ).append( "]" )
497                //      .append( CR )
498                        .append( "MINCOUNT=[" ).append( mincount ).append( "]," )
499                        .append( "MAXCOUNT=[" ).append( maxcount ).append( "]," )
500                        .append( "POOLTIME=[" ).append( pooltime ).append( "]," )
501                        .append( "READONLY=[" ).append( readonly ).append( "]," )
502                        .append( "APPINFO =[" ).append( isUseAppInfo ).append( ']' )            // 6.0.2.5 (2014/10/31) char を append する。
503                        .append( CR );
504                return buf.toString();
505        }
506}