170 lines
5.6 KiB
Factor
Executable File
170 lines
5.6 KiB
Factor
Executable File
! Copyright (C) 2008 Chris Double, Doug Coleman.
|
|
! See http://factorcode.org/license.txt for BSD license.
|
|
USING: alien.c-types arrays assocs kernel math math.parser
|
|
namespaces sequences db.sqlite.ffi db combinators
|
|
continuations db.types calendar.format serialize
|
|
io.streams.byte-array byte-arrays io.encodings.binary
|
|
io.backend db.errors ;
|
|
IN: db.sqlite.lib
|
|
|
|
ERROR: sqlite-error < db-error n string ;
|
|
ERROR: sqlite-sql-error < sql-error n string ;
|
|
|
|
: throw-sqlite-error ( n -- * )
|
|
dup sqlite-error-messages nth sqlite-error ;
|
|
|
|
: sqlite-statement-error ( -- * )
|
|
SQLITE_ERROR
|
|
db get db-handle sqlite3_errmsg sqlite-sql-error ;
|
|
|
|
: sqlite-check-result ( n -- )
|
|
{
|
|
{ SQLITE_OK [ ] }
|
|
{ SQLITE_ERROR [ sqlite-statement-error ] }
|
|
[ throw-sqlite-error ]
|
|
} case ;
|
|
|
|
: sqlite-open ( path -- db )
|
|
normalize-path
|
|
"void*" <c-object>
|
|
[ sqlite3_open sqlite-check-result ] keep *void* ;
|
|
|
|
: sqlite-close ( db -- )
|
|
sqlite3_close sqlite-check-result ;
|
|
|
|
: sqlite-prepare ( db sql -- handle )
|
|
dup length "void*" <c-object> "void*" <c-object>
|
|
[ sqlite3_prepare_v2 sqlite-check-result ] 2keep
|
|
drop *void* ;
|
|
|
|
: sqlite-bind-parameter-index ( handle name -- index )
|
|
sqlite3_bind_parameter_index ;
|
|
|
|
: parameter-index ( handle name text -- handle name text )
|
|
>r dupd sqlite-bind-parameter-index r> ;
|
|
|
|
: sqlite-bind-text ( handle index text -- )
|
|
dup length SQLITE_TRANSIENT
|
|
sqlite3_bind_text sqlite-check-result ;
|
|
|
|
: sqlite-bind-int ( handle i n -- )
|
|
sqlite3_bind_int sqlite-check-result ;
|
|
|
|
: sqlite-bind-int64 ( handle i n -- )
|
|
sqlite3_bind_int64 sqlite-check-result ;
|
|
|
|
: sqlite-bind-uint64 ( handle i n -- )
|
|
sqlite3-bind-uint64 sqlite-check-result ;
|
|
|
|
: sqlite-bind-double ( handle i x -- )
|
|
sqlite3_bind_double sqlite-check-result ;
|
|
|
|
: sqlite-bind-null ( handle i -- )
|
|
sqlite3_bind_null sqlite-check-result ;
|
|
|
|
: sqlite-bind-blob ( handle i byte-array -- )
|
|
dup length SQLITE_TRANSIENT
|
|
sqlite3_bind_blob sqlite-check-result ;
|
|
|
|
: sqlite-bind-text-by-name ( handle name text -- )
|
|
parameter-index sqlite-bind-text ;
|
|
|
|
: sqlite-bind-int-by-name ( handle name int -- )
|
|
parameter-index sqlite-bind-int ;
|
|
|
|
: sqlite-bind-int64-by-name ( handle name int64 -- )
|
|
parameter-index sqlite-bind-int64 ;
|
|
|
|
: sqlite-bind-uint64-by-name ( handle name int64 -- )
|
|
parameter-index sqlite-bind-uint64 ;
|
|
|
|
: sqlite-bind-double-by-name ( handle name double -- )
|
|
parameter-index sqlite-bind-double ;
|
|
|
|
: sqlite-bind-blob-by-name ( handle name blob -- )
|
|
parameter-index sqlite-bind-blob ;
|
|
|
|
: sqlite-bind-null-by-name ( handle name obj -- )
|
|
parameter-index drop sqlite-bind-null ;
|
|
|
|
: sqlite-bind-type ( handle key value type -- )
|
|
over [ drop NULL ] unless
|
|
dup array? [ first ] when
|
|
{
|
|
{ INTEGER [ sqlite-bind-int-by-name ] }
|
|
{ BIG-INTEGER [ sqlite-bind-int64-by-name ] }
|
|
{ SIGNED-BIG-INTEGER [ sqlite-bind-int64-by-name ] }
|
|
{ UNSIGNED-BIG-INTEGER [ sqlite-bind-uint64-by-name ] }
|
|
{ TEXT [ sqlite-bind-text-by-name ] }
|
|
{ VARCHAR [ sqlite-bind-text-by-name ] }
|
|
{ DOUBLE [ sqlite-bind-double-by-name ] }
|
|
{ DATE [ timestamp>ymd sqlite-bind-text-by-name ] }
|
|
{ TIME [ timestamp>hms sqlite-bind-text-by-name ] }
|
|
{ DATETIME [ timestamp>ymdhms sqlite-bind-text-by-name ] }
|
|
{ TIMESTAMP [ timestamp>ymdhms sqlite-bind-text-by-name ] }
|
|
{ BLOB [ sqlite-bind-blob-by-name ] }
|
|
{ FACTOR-BLOB [
|
|
object>bytes
|
|
sqlite-bind-blob-by-name
|
|
] }
|
|
{ +db-assigned-id+ [ sqlite-bind-int-by-name ] }
|
|
{ +random-id+ [ sqlite-bind-int64-by-name ] }
|
|
{ NULL [ sqlite-bind-null-by-name ] }
|
|
[ no-sql-type ]
|
|
} case ;
|
|
|
|
: sqlite-finalize ( handle -- ) sqlite3_finalize sqlite-check-result ;
|
|
: sqlite-reset ( handle -- ) sqlite3_reset sqlite-check-result ;
|
|
: sqlite-clear-bindings ( handle -- )
|
|
sqlite3_clear_bindings sqlite-check-result ;
|
|
: sqlite-#columns ( query -- int ) sqlite3_column_count ;
|
|
: sqlite-column ( handle index -- string ) sqlite3_column_text ;
|
|
: sqlite-column-name ( handle index -- string ) sqlite3_column_name ;
|
|
: sqlite-column-type ( handle index -- string ) sqlite3_column_type ;
|
|
|
|
: sqlite-column-blob ( handle index -- byte-array/f )
|
|
[ sqlite3_column_bytes ] 2keep
|
|
pick zero? [
|
|
3drop f
|
|
] [
|
|
sqlite3_column_blob swap memory>byte-array
|
|
] if ;
|
|
|
|
: sqlite-column-typed ( handle index type -- obj )
|
|
dup array? [ first ] when
|
|
{
|
|
{ +db-assigned-id+ [ sqlite3_column_int64 ] }
|
|
{ +random-id+ [ sqlite3-column-uint64 ] }
|
|
{ INTEGER [ sqlite3_column_int ] }
|
|
{ BIG-INTEGER [ sqlite3_column_int64 ] }
|
|
{ SIGNED-BIG-INTEGER [ sqlite3_column_int64 ] }
|
|
{ UNSIGNED-BIG-INTEGER [ sqlite3-column-uint64 ] }
|
|
{ DOUBLE [ sqlite3_column_double ] }
|
|
{ TEXT [ sqlite3_column_text ] }
|
|
{ VARCHAR [ sqlite3_column_text ] }
|
|
{ DATE [ sqlite3_column_text dup [ ymd>timestamp ] when ] }
|
|
{ TIME [ sqlite3_column_text dup [ hms>timestamp ] when ] }
|
|
{ TIMESTAMP [ sqlite3_column_text dup [ ymdhms>timestamp ] when ] }
|
|
{ DATETIME [ sqlite3_column_text dup [ ymdhms>timestamp ] when ] }
|
|
{ BLOB [ sqlite-column-blob ] }
|
|
{ FACTOR-BLOB [
|
|
sqlite-column-blob
|
|
dup [ bytes>object ] when
|
|
] }
|
|
! { NULL [ 2drop f ] }
|
|
[ no-sql-type ]
|
|
} case ;
|
|
|
|
: sqlite-row ( handle -- seq )
|
|
dup sqlite-#columns [ sqlite-column ] with map ;
|
|
|
|
: sqlite-step-has-more-rows? ( prepared -- bool )
|
|
{
|
|
{ SQLITE_ROW [ t ] }
|
|
{ SQLITE_DONE [ f ] }
|
|
[ sqlite-check-result f ]
|
|
} case ;
|
|
|
|
: sqlite-next ( prepared -- ? )
|
|
sqlite3_step sqlite-step-has-more-rows? ;
|