2008-02-03 16:06:31 -05:00
|
|
|
! Copyright (C) 2007, 2008 Doug Coleman.
|
2008-02-01 18:43:44 -05:00
|
|
|
! See http://factorcode.org/license.txt for BSD license.
|
|
|
|
USING: arrays assocs alien alien.syntax continuations io
|
2008-02-13 17:51:16 -05:00
|
|
|
kernel math math.parser namespaces prettyprint quotations
|
|
|
|
sequences debugger db db.postgresql.lib db.postgresql.ffi
|
2008-02-15 15:01:44 -05:00
|
|
|
db.tuples db.types tools.annotations math.ranges
|
2008-02-18 17:52:00 -05:00
|
|
|
combinators sequences.lib classes ;
|
2008-02-01 18:43:44 -05:00
|
|
|
IN: db.postgresql
|
|
|
|
|
|
|
|
TUPLE: postgresql-db host port pgopts pgtty db user pass ;
|
|
|
|
TUPLE: postgresql-statement ;
|
2008-02-03 16:06:31 -05:00
|
|
|
TUPLE: postgresql-result-set ;
|
2008-02-01 18:43:44 -05:00
|
|
|
: <postgresql-statement> ( statement -- postgresql-statement )
|
|
|
|
postgresql-statement construct-delegate ;
|
|
|
|
|
|
|
|
: <postgresql-db> ( host user pass db -- obj )
|
|
|
|
{
|
|
|
|
set-postgresql-db-host
|
|
|
|
set-postgresql-db-user
|
|
|
|
set-postgresql-db-pass
|
|
|
|
set-postgresql-db-db
|
|
|
|
} postgresql-db construct ;
|
|
|
|
|
|
|
|
M: postgresql-db db-open ( db -- )
|
|
|
|
dup {
|
|
|
|
postgresql-db-host
|
|
|
|
postgresql-db-port
|
|
|
|
postgresql-db-pgopts
|
|
|
|
postgresql-db-pgtty
|
|
|
|
postgresql-db-db
|
|
|
|
postgresql-db-user
|
|
|
|
postgresql-db-pass
|
|
|
|
} get-slots connect-postgres <db> swap set-delegate ;
|
|
|
|
|
|
|
|
M: postgresql-db dispose ( db -- )
|
|
|
|
db-handle PQfinish ;
|
|
|
|
|
|
|
|
: with-postgresql ( host ust pass db quot -- )
|
|
|
|
>r <postgresql-db> r> with-disposal ;
|
|
|
|
|
2008-02-03 16:06:31 -05:00
|
|
|
M: postgresql-statement bind-statement* ( seq statement -- )
|
|
|
|
set-statement-params ;
|
2008-02-03 00:28:33 -05:00
|
|
|
|
2008-02-12 18:19:55 -05:00
|
|
|
M: postgresql-statement reset-statement ( statement -- )
|
|
|
|
drop ;
|
2008-02-01 18:43:44 -05:00
|
|
|
|
2008-02-03 16:06:31 -05:00
|
|
|
M: postgresql-result-set #rows ( result-set -- n )
|
|
|
|
result-set-handle PQntuples ;
|
2008-02-01 18:43:44 -05:00
|
|
|
|
2008-02-03 16:06:31 -05:00
|
|
|
M: postgresql-result-set #columns ( result-set -- n )
|
|
|
|
result-set-handle PQnfields ;
|
2008-02-01 18:43:44 -05:00
|
|
|
|
2008-02-03 16:06:31 -05:00
|
|
|
M: postgresql-result-set row-column ( result-set n -- obj )
|
|
|
|
>r dup result-set-handle swap result-set-n r> PQgetvalue ;
|
2008-02-03 00:28:33 -05:00
|
|
|
|
2008-02-15 15:01:44 -05:00
|
|
|
M: postgresql-result-set row-column-typed ( result-set n type -- obj )
|
|
|
|
>r row-column r> sql-type>factor-type ;
|
|
|
|
|
|
|
|
M: postgresql-result-set sql-type>factor-type ( obj type -- newobj )
|
|
|
|
{
|
|
|
|
{ INTEGER [ string>number ] }
|
|
|
|
{ BIG_INTEGER [ string>number ] }
|
|
|
|
{ DOUBLE [ string>number ] }
|
|
|
|
[ drop ]
|
|
|
|
} case ;
|
2008-02-15 13:16:31 -05:00
|
|
|
|
2008-02-14 02:27:54 -05:00
|
|
|
M: postgresql-statement insert-statement ( statement -- id )
|
2008-02-15 21:37:54 -05:00
|
|
|
query-results [ 0 row-column ] with-disposal string>number ;
|
2008-02-03 00:28:33 -05:00
|
|
|
|
2008-02-03 16:06:31 -05:00
|
|
|
M: postgresql-statement query-results ( query -- result-set )
|
|
|
|
dup statement-params [
|
|
|
|
over [ bind-statement ] keep
|
|
|
|
do-postgresql-bound-statement
|
|
|
|
] [
|
|
|
|
dup do-postgresql-statement
|
|
|
|
] if*
|
|
|
|
postgresql-result-set <result-set>
|
|
|
|
dup init-result-set ;
|
|
|
|
|
2008-02-15 00:39:20 -05:00
|
|
|
M: postgresql-result-set advance-row ( result-set -- )
|
|
|
|
dup result-set-n 1+ swap set-result-set-n ;
|
|
|
|
|
|
|
|
M: postgresql-result-set more-rows? ( result-set -- ? )
|
|
|
|
dup result-set-n swap result-set-max < ;
|
2008-02-01 18:43:44 -05:00
|
|
|
|
|
|
|
M: postgresql-statement dispose ( query -- )
|
|
|
|
dup statement-handle PQclear
|
2008-02-03 00:28:33 -05:00
|
|
|
f swap set-statement-handle ;
|
|
|
|
|
|
|
|
M: postgresql-result-set dispose ( result-set -- )
|
|
|
|
dup result-set-handle PQclear
|
|
|
|
0 0 f roll {
|
2008-02-03 16:06:31 -05:00
|
|
|
set-result-set-n set-result-set-max set-result-set-handle
|
2008-02-03 00:28:33 -05:00
|
|
|
} set-slots ;
|
2008-02-01 18:43:44 -05:00
|
|
|
|
|
|
|
M: postgresql-statement prepare-statement ( statement -- )
|
|
|
|
[
|
|
|
|
>r db get db-handle "" r>
|
|
|
|
dup statement-sql swap statement-params
|
2008-02-03 16:06:31 -05:00
|
|
|
length f PQprepare postgresql-error
|
2008-02-01 18:43:44 -05:00
|
|
|
] keep set-statement-handle ;
|
|
|
|
|
|
|
|
M: postgresql-db <simple-statement> ( sql -- statement )
|
|
|
|
{ set-statement-sql } statement construct
|
|
|
|
<postgresql-statement> ;
|
|
|
|
|
|
|
|
M: postgresql-db <prepared-statement> ( sql -- statement )
|
2008-02-03 00:28:33 -05:00
|
|
|
{ set-statement-sql } statement construct
|
|
|
|
<postgresql-statement> ;
|
2008-02-03 16:06:31 -05:00
|
|
|
|
|
|
|
M: postgresql-db begin-transaction ( -- )
|
|
|
|
"BEGIN" sql-command ;
|
|
|
|
|
|
|
|
M: postgresql-db commit-transaction ( -- )
|
|
|
|
"COMMIT" sql-command ;
|
|
|
|
|
|
|
|
M: postgresql-db rollback-transaction ( -- )
|
|
|
|
"ROLLBACK" sql-command ;
|
2008-02-13 17:51:16 -05:00
|
|
|
|
2008-02-18 17:52:00 -05:00
|
|
|
: insert-function ( columns table -- sql types )
|
2008-02-13 17:51:16 -05:00
|
|
|
[
|
2008-02-15 00:39:20 -05:00
|
|
|
>r remove-id r>
|
2008-02-14 02:27:54 -05:00
|
|
|
"create function add_" % dup %
|
|
|
|
"(" %
|
|
|
|
over [ "," % ]
|
2008-02-18 17:52:00 -05:00
|
|
|
[ third dup array? [ first ] when >sql-type-string % ] interleave
|
2008-02-14 02:27:54 -05:00
|
|
|
")" %
|
|
|
|
" returns bigint as '" %
|
2008-02-13 17:51:16 -05:00
|
|
|
|
2008-02-14 02:27:54 -05:00
|
|
|
2dup "insert into " %
|
2008-02-13 17:51:16 -05:00
|
|
|
%
|
|
|
|
"(" %
|
|
|
|
dup [ ", " % ] [ second % ] interleave
|
|
|
|
") " %
|
|
|
|
" values (" %
|
2008-02-15 00:39:20 -05:00
|
|
|
length [1,b] [ ", " % ] [ "$" % # ] interleave
|
2008-02-14 02:27:54 -05:00
|
|
|
"); " %
|
|
|
|
|
|
|
|
"select currval(''" % % "_id_seq'');' language sql;" %
|
|
|
|
drop
|
2008-02-18 17:52:00 -05:00
|
|
|
] "" make f ;
|
2008-02-14 02:27:54 -05:00
|
|
|
|
2008-02-15 00:39:20 -05:00
|
|
|
: drop-function ( columns table -- sql )
|
2008-02-14 02:27:54 -05:00
|
|
|
[
|
2008-02-15 00:39:20 -05:00
|
|
|
>r remove-id r>
|
|
|
|
"drop function add_" % %
|
2008-02-14 02:27:54 -05:00
|
|
|
"(" %
|
2008-02-18 17:52:00 -05:00
|
|
|
[ "," % ] [ third >sql-type-string % ] interleave
|
2008-02-13 17:51:16 -05:00
|
|
|
")" %
|
2008-02-18 17:52:00 -05:00
|
|
|
] "" nmake ;
|
|
|
|
|
|
|
|
! M: postgresql-db create-sql ( columns table -- seq )
|
|
|
|
! [
|
|
|
|
! [
|
|
|
|
! 2dup
|
|
|
|
! "create table " % %
|
|
|
|
! " (" % [ ", " % ] [
|
|
|
|
! dup second % " " %
|
|
|
|
! dup third >sql-type-string % " " %
|
|
|
|
! sql-modifiers " " join %
|
|
|
|
! ] interleave "); " %
|
|
|
|
! ] "" make ,
|
|
|
|
!
|
|
|
|
! over native-id? [ insert-function , ] [ 2drop ] if
|
|
|
|
! ] { } make ;
|
2008-02-15 00:39:20 -05:00
|
|
|
|
|
|
|
M: postgresql-db drop-sql ( columns table -- seq )
|
|
|
|
[
|
|
|
|
[
|
|
|
|
dup "drop table " % % ";" %
|
|
|
|
] "" make ,
|
|
|
|
over native-id? [ drop-function , ] [ 2drop ] if
|
|
|
|
] { } make ;
|
2008-02-14 02:27:54 -05:00
|
|
|
|
2008-02-18 17:52:00 -05:00
|
|
|
M: postgresql-db insert-sql* ( columns table -- sql slots )
|
2008-02-14 02:27:54 -05:00
|
|
|
[
|
|
|
|
"select add_" % %
|
|
|
|
"(" %
|
2008-02-18 17:52:00 -05:00
|
|
|
dup length [1,b] [ ", " % ] [ "$" % # ] interleave
|
|
|
|
");" %
|
2008-02-15 00:39:20 -05:00
|
|
|
] "" make ;
|
2008-02-14 02:27:54 -05:00
|
|
|
|
2008-02-18 17:52:00 -05:00
|
|
|
M: postgresql-db update-sql* ( columns table -- sql slots )
|
2008-02-13 17:51:16 -05:00
|
|
|
[
|
|
|
|
"update " %
|
|
|
|
%
|
|
|
|
" set " %
|
|
|
|
dup remove-id
|
2008-02-15 00:39:20 -05:00
|
|
|
dup length [1,b] swap 2array flip
|
|
|
|
[ ", " % ] [ first2 second % " = $" % # ] interleave
|
2008-02-13 17:51:16 -05:00
|
|
|
" where " %
|
2008-02-15 00:39:20 -05:00
|
|
|
[ primary-key? ] find nip second dup % " = $" % length 2 + #
|
|
|
|
] "" make ;
|
2008-02-13 17:51:16 -05:00
|
|
|
|
2008-02-15 21:37:54 -05:00
|
|
|
M: postgresql-db delete-sql* ( columns table -- slot-names sql )
|
2008-02-13 17:51:16 -05:00
|
|
|
[
|
|
|
|
"delete from " %
|
|
|
|
%
|
|
|
|
" where " %
|
2008-02-15 00:39:20 -05:00
|
|
|
first second % " = $1" %
|
|
|
|
] "" make ;
|
2008-02-13 17:51:16 -05:00
|
|
|
|
2008-02-18 17:52:00 -05:00
|
|
|
: column-name% ( spec -- )
|
|
|
|
dup sql-spec-column-name 0%
|
|
|
|
sql-spec-type >sql-type-string 1, ;
|
|
|
|
|
|
|
|
: column-names% ( class -- )
|
|
|
|
db-columns [ "," 0, ] [ column-name% ] interleave ;
|
|
|
|
|
|
|
|
M: postgresql-db column-bind% ( spec -- )
|
|
|
|
|
|
|
|
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
|
|
! : select-foreign-table-sql ( tuple relation -- )
|
|
|
|
! ! select id, name, age from puppy, basket where puppy.basket_id = basket.id
|
|
|
|
! "select " 0%
|
|
|
|
! ;
|
|
|
|
! TODO
|
|
|
|
: select-relations-sql ( tuple -- seq )
|
|
|
|
! seq -- { sql types }
|
|
|
|
dup class db-relations [
|
|
|
|
[
|
|
|
|
! select-foreign-table-sql
|
|
|
|
] { "" { } } 2 nmake
|
|
|
|
] with { } map>assoc ;
|
|
|
|
|
|
|
|
! TODO
|
|
|
|
: select-by-slots-sql ( tuple -- sql )
|
|
|
|
dup tuple>filled-slots
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
|
|
M: postgresql-db select-sql ( tuple -- sql slot-names )
|
|
|
|
[
|
|
|
|
|
|
|
|
] { } 2 nmake ;
|
2008-02-13 17:51:16 -05:00
|
|
|
|
|
|
|
M: postgresql-db tuple>params ( columns tuple -- obj )
|
2008-02-14 02:27:54 -05:00
|
|
|
[ >r dup third swap first r> get-slot-named swap ]
|
|
|
|
curry { } map>assoc ;
|
2008-02-13 17:51:16 -05:00
|
|
|
|
|
|
|
: postgresql-db-modifiers ( -- hashtable )
|
|
|
|
H{
|
2008-02-18 17:52:00 -05:00
|
|
|
{ +native-id+ "primary key" }
|
|
|
|
{ +foreign-key+ "" }
|
2008-02-13 17:51:16 -05:00
|
|
|
{ +assigned-id+ "primary key" }
|
|
|
|
{ +autoincrement+ "autoincrement" }
|
|
|
|
{ +unique+ "unique" }
|
|
|
|
{ +default+ "default" }
|
|
|
|
{ +null+ "null" }
|
|
|
|
{ +not-null+ "not null" }
|
|
|
|
} ;
|
|
|
|
|
2008-02-18 17:52:00 -05:00
|
|
|
! M: postgresql-db sql-modifier>string ( modifier -- str )
|
|
|
|
! dup array? [
|
|
|
|
! first2
|
|
|
|
! >r swap at r> number>string*
|
|
|
|
! " " swap 3append
|
|
|
|
! ] [
|
|
|
|
! swap at
|
|
|
|
! ] if ;
|
|
|
|
!
|
|
|
|
! M: postgresql-db sql-modifiers* ( modifiers -- str )
|
|
|
|
! postgresql-db-modifiers swap [
|
|
|
|
! sql-modifier>string
|
|
|
|
! ] with map [ ] subset ;
|