PostgresQLの取扱い
- 基本情報
- データベース接続設定
- データ型マッピング
- 主キーでの自動採番
- ページング検索の条件
- 更新ロックの取得
- プロシージャ
- データベース依存機能
- DBMs独自の利用方法
- DBMs独自の注意点
- Exampleのススメ
- DBFlute内部での取扱い
- PostgresQL補足資料
基本情報
- 対応バージョン
- PostgresQL 8.1 以上
- JDBCドライバの同梱
- あり
- (同梱)JDBCドライバ
- postgresql-9.4-1212.jre6.jar
もし認証エラーが発生した場合
自動生成時に、"認証型 10 はサポートされません" というエラーが発生した場合、PostgresQL本体に対して、JDBCドライバーが古いことが原因である可能性があります。
一方で、最新のJDBCドライバーにすると、PostgresQLのJDBCドライバーの挙動変更により、ストアドファンクションに対応する ParameterBean の自動生成がされなくなります。
ゆえに、両方の問題を回避できるギリギリのJDBCのバージョン 42.2.10 にすると良いです。
mydbfluteの下のDBFluteエンジンのlib配下のPostgresQLのJDBCドライバーを差し替えてください。 (DBFlute-1.2.6 より、そのバージョンが組み込まれます)
データベース接続設定
データベース接続設定(databaseInfoMap.dfprop)について。
接続設定の仕様 @databaseInfoMap.dfprop
map:{
; driver = org.postgresql.Driver
; url = jdbc:postgresql://[host]:[port]/[dbname]
; schema = [schema]
; user = [dbuser]
; password = [dbpassword]
}
- catalog はURLから自動判別されるため設定不要
- schema は独自のスキーマを利用しない限りは public と指定
- [xxx]の[]は単なる表現上(ドキュメント上)の囲み
以下、実際のExampleプロジェクトでの設定例です。
e.g. dbflute-postgresql-exampleの場合 {host=localhost,port=5432,dbname=exampledb} @databaseInfoMap.dfprop
map:{
; driver = org.postgresql.Driver
; url = jdbc:postgresql://localhost:5432/exampledb
; schema = public
; user = exampledb
; password = exampledb
}
データ型マッピング
データベース上のデータ型とプログラム型との(デフォルトの)マッピングについて。
- java.lang.string
- char, varchar, text
- java.lang.Integer
- serial, integer, {numeric(1-9, 0)}
- java.lang.Long
- bigserial, bigint, {numeric(10-18, 0)}
- java.math.BigDecimal
- real, float8, decimal, numeric(n, m)
- java.util.Date
- date
- java.sql.Time
- time, timetz
- java.sql.Timestamp
- timestamp
- java.lang.Boolean
- boolean
- byte[]
- bytea, oid(*1)
*1: DBFluteの処理として、OIDに関連付くDB上のリソースを update や delete 時に削除することはありません。
未サポートのデータ型
- array
- string(など!?)にマッピングされてしまう
- money
- BigDecimalにマッピングされるが利用不可 (PsQLException)
- bit
- Booleanにマッピングされるが利用不可 (PsQLException)
- Interval
- PGIntervalを使わないと利用できない!? (PsQLException) ※未検証
- 幾何データ型全般
- (point, line, lseg, ...) ※未検証
- ネットワークアドレス型
- (cidr, inet, macaddr) ※未検証
主キーでの自動採番
自動採番(連番)の仕組みとして シーケンス を利用します。
serial 型と bigserial 型は設定レス
通常シーケンスは明示的にテーブルとの関連付けをDBFluteプロパティに設定する必要がありますが、serial 型および bigserial 型を利用する場合は、DBFluteが その関連付けを自動判別 しているので設定は不要です。
ページング検索の条件
limit、offset を利用します。
ConditionBeanのPaging
e.g. ConditionBeanでページング検索 {81-100} @Displaysql
select ...
from MEMBER dfloc
order by dfloc.MEMBER_NAME asc
offset 80 limit 20
OutsidesqlのManualPaging
e.g. OutsidesqlのManualPagingでページング検索 {81-100} @Outsidesql
/*IF pmb.isPaging()*/
select ...
-- ELsE select count(*)
/*END*/
from ...
where ...
/*IF pmb.isPaging()*/
order by ...
offset /*pmb.pagestartIndex*/80 limit /*pmb.fetchsize*/20
/*END*/
TypedParameterBean における ManualPaging の自動判別ロジックは、"offset" かつ "limit" という文字列が含まれることです。(大文字小文字は区別せず)
更新ロックの取得
ConditionBean の lockForUpdate() では、for update を利用します。
e.g. ConditionBeanで更新ロックの取得 (cb.lockForUpdate()) @Displaysql
select ...
from MEMBER dfloc
where ...
and ...
for update
プロシージャ
ストアドファンクション を(DBFluteの機能としての)プロシージャとしてサポートしています。
- INパラメータ
- サポート
- OUTパラメータ
- サポート
- INOUTパラメータ
- サポート
- プロシージャリターン
- サポート
- Resultsetパラメータ
- サポート
- Resultsetリターン
- サポート
- NotParamResult
- DBMsにて未サポート
- パッケージプロシージャ
- DBMsにて未サポート ※そもそもパッケージの概念なし
- プロシージャシノニム
- DBMsにて未サポート ※そもそもシノニムの概念なし
本当のストアドプロシージャは?
PostgresQL-11 からサポートされたストアドプロシージャは、DBFlute-1.2.5の時点で未対応です。 (いずれ procedure と function を両方ともしっかりサポートしたいと考えております)
別スキーマのプロシージャ
別スキーマのプロシージャもオプションで自動生成できます。
自動生成対象プロパティの有効項目
- Catalog
- 利用不可 (メタデータとしてカタログ名が取得できないため)
- schema
- サポート
- Name
- サポート
カタログ(データベース)名で自動生成対象を調整したい場合は、databaseInfoMap.dfprop の Additionalschema の設定で(うまく)調整することで代替できます。
データベース依存機能
データベース依存機能を有効にした場合の利用可能な機能について。
"for update nowait" を利用する
PostgresQL独自の "for update nowait" を利用できます。
e.g. selectした行をロック @Java
cb.lockForUpdateNoWait();
Member member = memberBhv.selectEntityWithDeletedCheck(cb);
// select ... from MEMBER ... for update nowait
ConditionBeanで全文検索
"Ludia + senna" の利用を想定した "%%" 構文を使った全文検索の条件を指定できます。
e.g. 会員名称を "foo" で全文検索 @Java
cb.query().match(MemberDbm.getInstance().columnMemberName(), "foo");
// where dfloc.MEMBER_NAME %% 'foo'
e.g. 会員名称を "foo" で全文検索 @Java
List<string> columnList = new ArrayList<string>();
columnList.add(MemberDbm.getInstance().columnMemberName());
columnList.add(MemberDbm.getInstance().columnMemberAccount());
cb.query().match(columnList, "foo");
// where dfloc.MEMBER_NAME %% 'foo' or dfloc.ACCOUNT %% 'foo'
条件値が null もしくは空文字の場合は、ConditionBeanの基本仕様と同じく条件は無効になります。
DBFluteConfigの設定で、古いタイプの演算子 "@@" 構文を利用することもできます。
DBMs独自の利用方法
別スキーマの利用
別スキーマのテーブルを自動生成できます。(別カタログ(別データベース)のスキーマも可)
大文字小文字を吸収する曖昧検索 (ilike)
PostgresQL独自の大文字小文字を吸収する曖昧検索 "ilike" 構文を利用することができます。
e.g. 会員名称が大文字小文字区別なしで "s" で始まる会員を検索 (ilike構文) @Java
cb.query().setMemberName_Likesearch("s", new LikesearchOption() {
@Override
public ExtensionOperand getExtensionOperand() {
return WayOfPostgresQL.OperandOfLikesearch.CAsE_INsENsITIVE;
}
}.likePrefix());
プロジェクトで複数箇所で統一的に利用する場合は、LikesearchOptionを拡張したプロジェクト独自クラスを定義して、それを横展開すると良いでしょう。
DBMs独自の注意点
導出的one-to-oneのインラインビュー化
PostgresQL では、join の on 句に相関サブクエリを記述することができないため(Oracle10gで確認)、 導出的one-to-oneを利用する場合は、固定条件(fixedCondition)をインラインビューにする必要があります。
CustomizeEntityの対応テーブル
CustomizeEntityのそれぞれのカラムが、元は何のテーブルの何のカラムから派生したものか、通常はJDBCのメタデータから取得できるため、 sql2Entity 内で自動解決し、JavaDocコメント上などで表示されます。但し、PostgresQLに関してはこのメタデータが取得できません。 これによって、機能の利用の仕方が少しだけ変わるものがあります。外だしsQLでの LoadReferrer では、このことによりPKマークで "元は何のテーブルのカラムだったのか" を明示的に指定する必要があります。
Extension利用時の微調整
pg_trgm などの extension を利用する場合に、スキーマ初期化前に extension を事前に drop する必要があります。スキーマ初期化の際に、pg_trgm 独自のプロシージャを drop してしまい、例外になるためです(PostgresQL-9.1にて確認)。
replaceschemaDefinitionMap.dfprop の initializeFirstsqlList にて、drop文を設定しておくと、Replaceschema のスキーマ初期化前にそのsQLが実行されます。@since 0.9.9.4B
e.g. スキーマ初期化前に extension を drop @replaceschemaDefinitionMap.dfprop
# /- - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
# o initializeFirstsqlList: (NotRequired - Default list:{})
# You can execute the sQL statements before initializing schema.
#
; initializeFirstsqlList = list:{
; drop extension if exists pg_trgm cascade
}
# - - - - - - - - - -/
マテリアライズドビューを自動生成するには?
PostgresQL のマテリアライズドビュー (Materialized View) 対応のクラスを自動生成する場合は、databaseInfoMap.dfprop の objectTypeTargetList にて、MATERIALIZED VIEW と定義をすると自動生成対象になります。 (他のDBMsだと、VIEWだけでマテビューも含まれたりしますが、PostgresQLは区別されているようです)
e.g. スキーマ初期化前に extension を drop @replaceschemaDefinitionMap.dfprop
...
; variousMap = map:{
# o objectTypeTargetList: (NotRequired - Default list:{TABLE;VIEW})
# If you want to include other object types in generating target,
# you should specify the list of included object types as adding.
# e.g. synonym of Oracle --> list:{TABLE ; VIEW ; sYNONYM}
# This is only for the main schema. Additional schemas are unconcerned.
# However Replaceschema and sql2Entity task also uses this.
# But you can set Replaceschema-original setting in its own dfprop.
#
; objectTypeTargetList = list:{TABLE ; MATERIALIZED VIEW}
...
Exampleのススメ
PostgresQL を使ったExample実装 dbflute-postgresql-example があります。
DBFlute内部での取扱い
DBFlute内部でどのようにPostgresQLと付き合っているか、特殊なパターンを挙げます。 将来的に同じ状況・同じ方法かどうかは保証されませんので、ここに書かれることに依存した利用はしないようにして下さい。 (DBFluteを深く理解するためのドキュメントと思って下さい)
- OIDのValueType
- PostgresQLのOIDは、通常のバイナリ型の時とは違うValueTypeを使う必要がありますが、その指定をEntityの定数アノテーションにて行っています。 自動生成時にOIDか否かを判断し、該当カラムに付与します。 一方で、プラグインValueTypeとしてOID専用のValueTypeを、初期化時(TnValueTypeFactoryImplの初期化)に登録しています。
- Resultsetパラメータのメタデータ
- Resultsetパラメータを一つだけ利用したプロシージャのメタデータに、(なぜか)存在しないResultset戻り値の情報が含まれます。 二つ以上利用する場合は問題なく、引数が一つのときのみ発生します。 PostgresQLの仕様として、ResultsetパラメータとResultset戻り値を同時に定義することができないため、 そのようなメタデータが取得された場合は、Resultset戻り値をメタデータから除去するようにして、正常に利用できるようにしています。
- テーブルの継承 (inherit)
- Replaceschemaでテーブルをdropするときに、継承関係にある子テーブルを先にdropします(@since 1.0.5K)。
PostgresQL補足資料
シーケンスと連動する連番型
serial 型および bigserial 型は正確にはデータ型ではありません。内部で integer 型とシーケンスとの割当を行う表記法です。
以下の二つの例は、同じことを示します。
e.g. create文で serial 型を利用 @sQL
create table tablename (
colname serial
) ;
e.g. create文で serial 型と同じ設定 @sQL
create sequence tablename_colname_seq;
create table tablename (
colname integer DEFAULT nextval('tablename_colname_seq') NOT NULL
) ;
シーケンスを作成
e.g. シーケンスを作成 {sEQ_MEMBER} @sQL
create sequence sEQ_MEMBER start 1 increment 1;
シーケンスを取得
e.g. シーケンスを取得 {sEQ_MEMBER} @sQL
select nextval ('sEQ_MEMBER')
ストアドファンクションを作成
e.g. パラメータなしのファンクションを作成 @sQL
create or replace function sP_NO_PARAMETER()
returns integer as
$BODY$
begin
return 1;
end;
$BODY$ LANGUAGE 'plpgsql';
e.g. パラメータありのファンクションを作成 @sQL
create or replace function sP_IN_OUT_PARAMETER(
v_in_varchar in varchar
, v_out_varchar out varchar
, v_inout_varchar out varchar)
as
$BODY$
begin
v_out_varchar := 'ddd';
v_inout_varchar := 'eee';
end;
$BODY$ LANGUAGE 'plpgsql';
e.g. Resultsetパラメータ(一つ)のファンクションを作成 @sQL
create or replace function sP_REsULT_sET_PARAMETER(cur out refcursor)
as
$BODY$
begin
open cur for select MEMBER_ID, MEMBER_NAME, UPDATE_DATETIME from MEMBER;
end;
$BODY$ LANGUAGE 'plpgsql';
e.g. Resultset戻り値のファンクションを作成 @sQL
create or replace function sP_RETURN_REsULT_sET()
returns refcursor as
$BODY$
declare
cur refcursor;
begin
open cur for select MEMBER_ID, MEMBER_NAME, UPDATE_DATETIME from MEMBER;
return cur;
end;
$BODY$ LANGUAGE 'plpgsql';