|  | # dio.tcl -- implements a database abstraction layer. | 
|  |  | 
|  | # Copyright 2002-2004 The Apache Software Foundation | 
|  | # | 
|  | # Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | # you may not use this file except in compliance with the License. | 
|  | # You may obtain a copy of the License at | 
|  | # | 
|  | #       http://www.apache.org/licenses/LICENSE-2.0 | 
|  | # | 
|  | # Unless required by applicable law or agreed to in writing, software | 
|  | # distributed under the License is distributed on an "AS IS" BASIS, | 
|  | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | # See the License for the specific language governing permissions and | 
|  | # limitations under the License. | 
|  |  | 
|  | catch {package require Tclx} | 
|  | package require Itcl | 
|  | package require dio::formatters | 
|  |  | 
|  | # Command ::rivet::lempty is extensively used within this class but it's | 
|  | # defined only when we run DIO from mod_rivet. For convenience we load it | 
|  | # here if needed | 
|  |  | 
|  | if {[info commands ::rivet::lempty] == ""} { | 
|  |  | 
|  | namespace eval ::rivet { | 
|  |  | 
|  | proc lempty {list} { | 
|  | if {[catch {llength $list} len]} { return 0 } | 
|  | return [expr {$len == 0}] | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | ##set auto_path [linsert $auto_path 0 [file dirname [info script]]] | 
|  |  | 
|  | namespace eval ::DIO { | 
|  |  | 
|  | proc handle {interface args} { | 
|  | set obj ::DIO::#auto | 
|  | set tdbc_driver "" | 
|  | if {($interface == "Tdbc") && ([llength $args] > 0)} { | 
|  | set tdbc_driver [lindex $args 0] | 
|  | set args [lreplace $args 0 0] | 
|  | } | 
|  |  | 
|  | #puts "interface: $interface ($tdbc_driver)" | 
|  | set first [lindex $args 0] | 
|  | if {![::rivet::lempty $first] && [string index $first 0] != "-"} { | 
|  | set obj  [lindex $args 0] | 
|  | set args [lreplace $args 0 0] | 
|  | } | 
|  | uplevel \#0 package require dio_$interface | 
|  |  | 
|  | #puts "tdbc: '$tdbc_driver' obj: '$obj' args 3: '$args'" | 
|  | if {$tdbc_driver == ""} { | 
|  |  | 
|  | # old connectors based on traditional dbms drivers | 
|  |  | 
|  | set dio_o [uplevel \#0 ::DIO::$interface $obj $args] | 
|  | } else { | 
|  | set dio_o [uplevel \#0 ::DIO::$interface $obj $tdbc_driver {*}$args] | 
|  | } | 
|  |  | 
|  | # the Tdbc driver may rename the drive because we created | 
|  | # a set of equivalence classes among DBMS (eg mariadb,sqlite etc) | 
|  | # therefore the class for the interface passed by the constructor | 
|  | # may not exist and it's ::DIO::Tdbc responsability to create it | 
|  |  | 
|  | $dio_o create_field_formatter | 
|  | return $dio_o | 
|  | } | 
|  |  | 
|  | ## | 
|  | # DATABASE CLASS | 
|  | ## | 
|  | ::itcl::class Database { | 
|  | constructor {args} { | 
|  | eval configure $args | 
|  | } | 
|  |  | 
|  | destructor { | 
|  | close | 
|  | } | 
|  |  | 
|  | # | 
|  | # result - generate a new DIO result object for the specified database | 
|  | # interface, with key-value pairs that get configured into the new | 
|  | # result object. | 
|  | # | 
|  | protected method result {interface args} { | 
|  | return [eval uplevel \#0 ::DIO::${interface}Result \#auto $args] | 
|  | } | 
|  |  | 
|  | # | 
|  | # quote - given a string, return the same string with any single | 
|  | #  quote characters preceded by a backslash | 
|  | # | 
|  | method quote {a_string} { | 
|  | return [DIO::formatters::quote $a_string] | 
|  | } | 
|  |  | 
|  | # | 
|  | # build_select_query - build a select query based on given arguments, | 
|  | #  which can include a table name, a select statement, switches to | 
|  | # turn on boolean AND or OR processing, and possibly | 
|  | # some key-value pairs that cause the where clause to be | 
|  | # generated accordingly | 
|  | # | 
|  | protected method build_select_query {args} { | 
|  |  | 
|  | set bool    AND | 
|  | set first   1 | 
|  | set req     "" | 
|  | set myTable $table | 
|  | set what    "*" | 
|  |  | 
|  | # for each argument passed us... | 
|  | # (we go by integers because we mess with the index based on | 
|  | #  what we find) | 
|  | for {set i 0} {$i < [llength $args]} {incr i} { | 
|  | # fetch the argument we're currently processing | 
|  | set elem [lindex $args $i] | 
|  |  | 
|  | switch -- [::string tolower $elem] { | 
|  | "-and" { | 
|  | # -and -- switch to AND-style processing | 
|  | set bool AND | 
|  | } | 
|  |  | 
|  | "-or"  { | 
|  | # -or -- switch to OR-style processing | 
|  | set bool OR | 
|  | } | 
|  |  | 
|  | "-table" { | 
|  | # -table -- identify which table the query is about | 
|  | set myTable [lindex $args [incr i]] | 
|  | } | 
|  |  | 
|  | "-select" { | 
|  | # -select - | 
|  | set what [lindex $args [incr i]] | 
|  | } | 
|  |  | 
|  | default { | 
|  | # it wasn't -and, -or, -table, or -select... | 
|  |  | 
|  | # if the first character of the element is a dash, | 
|  | # it's a field name and a value | 
|  |  | 
|  | if {[::string index $elem 0] == "-"} { | 
|  | set field [::string range $elem 1 end] | 
|  | set elem [lindex $args [incr i]] | 
|  |  | 
|  | # if it's the first field being processed, append | 
|  | # WHERE to the SQL request we're generating | 
|  | if {$first} { | 
|  | append req " WHERE" | 
|  | set first 0 | 
|  | } else { | 
|  | # it's not the first variable in the comparison | 
|  | # expression, so append the boolean state, either | 
|  | # AND or OR | 
|  | append req " $bool" | 
|  | } | 
|  |  | 
|  | # convert any asterisks to percent signs in the | 
|  | # value field | 
|  | regsub -all {\*} $elem {%} elem | 
|  |  | 
|  | # if there is a percent sign in the value | 
|  | # field now (having been there originally or | 
|  | # mapped in there a moment ago),  the SQL aspect | 
|  | # is appended with a "field LIKE value" | 
|  |  | 
|  | if {[::string first {%} $elem] != -1} { | 
|  | append req " $field LIKE [makeDBFieldValue $myTable $field $elem]" | 
|  | } elseif {[regexp {^([<>]) *([0-9.]*)$} $elem _ fn val]} { | 
|  | # value starts with <, or >, then space, | 
|  | # and a something | 
|  | append req " $field$fn$val" | 
|  | } elseif {[regexp {^([<>]=) *([0-9.]*)$} $elem _ fn val]} { | 
|  | # value starts with <= or >=, space, and something. | 
|  | append req " $field$fn$val" | 
|  | } else { | 
|  | # otherwise it's a straight key=value comparison | 
|  | append req " $field=[makeDBFieldValue $myTable $field $elem]" | 
|  | } | 
|  |  | 
|  | continue | 
|  | } | 
|  | append req " $elem" | 
|  | } | 
|  | } | 
|  | } | 
|  | return "select $what from $myTable $req" | 
|  | } | 
|  |  | 
|  | # | 
|  | # build_insert_query -- given an array name, a list of fields, and | 
|  | # possibly a table name, return a SQL insert statement inserting | 
|  | # into the named table (or the object's table variable, if none | 
|  | # is specified) for all of the fields specified, with their values | 
|  | # coming from the array | 
|  | # | 
|  | protected method build_insert_query {arrayName fields {myTable ""}} { | 
|  | upvar 1 $arrayName array | 
|  |  | 
|  | if {[::rivet::lempty $myTable]} { set myTable $table } | 
|  | set vals [::list] | 
|  | set vars [::list] | 
|  | foreach field $fields { | 
|  | if {![info exists array($field)]} { continue } | 
|  | lappend vars "$field" | 
|  | lappend vals "[makeDBFieldValue $myTable $field $array($field)]" | 
|  | } | 
|  | return "insert into $myTable ([join $vars {,}]) VALUES ([join $vals {,}])" | 
|  | } | 
|  |  | 
|  | # | 
|  | # build_update_query -- given an array name, a list of fields, and | 
|  | # possibly a table name, return a SQL update statement updating | 
|  | # the named table (or using object's table variable, if none | 
|  | # is named) for all of the fields specified, with their values | 
|  | # coming from the array | 
|  | # | 
|  | # note that after use a where clause still neds to be added or | 
|  | # you might update a lot more than you bargained for | 
|  | # | 
|  | protected method build_update_query {arrayName fields {myTable ""}} { | 
|  | upvar 1 $arrayName array | 
|  | if {[::rivet::lempty $myTable]} { set myTable $table } | 
|  | set string [::list] | 
|  | foreach field $fields { | 
|  | if {![info exists array($field)]} { continue } | 
|  | lappend string "$field=[makeDBFieldValue $myTable $field $array($field)]" | 
|  | } | 
|  |  | 
|  | return "UPDATE $myTable SET [join $string {,}]" | 
|  | } | 
|  |  | 
|  | # | 
|  | # lassign_array - given a list, an array name, and a variable number | 
|  | # of arguments consisting of variable names, assign each element in | 
|  | # the list, in turn, to elements corresponding to the variable | 
|  | # arguments, into the named array.  From TclX. | 
|  | # | 
|  | protected method lassign_array {list arrayName args} { | 
|  | upvar 1 $arrayName array | 
|  | foreach elem $list field $args { | 
|  | set array($field) $elem | 
|  | } | 
|  | } | 
|  |  | 
|  | # | 
|  | # configure_variable - given a variable name and a string, if the | 
|  | # string is empty return the variable name, otherwise set the | 
|  | # variable to the string. | 
|  | # | 
|  | protected method configure_variable {varName string} { | 
|  | if {[::rivet::lempty $string]} { return [cget -$varName] } | 
|  | configure -$varName $string | 
|  | } | 
|  |  | 
|  | # | 
|  | # build_where_key_clause - given a list of one or more key fields and | 
|  | # a corresponding list of one or more key values, construct a | 
|  | # SQL where clause that boolean ANDs all of the key-value pairs | 
|  | # together. | 
|  | # | 
|  | protected method build_key_where_clause {myKeyfield myKey} { | 
|  | ## If we're not using multiple keyfields, just return a simple | 
|  | ## where clause. | 
|  | if {[llength $myKeyfield] < 2} { | 
|  | return " WHERE $myKeyfield = [makeDBFieldValue $table $myKeyfield $myKey]" | 
|  | } | 
|  |  | 
|  | # multiple fields, construct it as a where-and | 
|  | set req " WHERE 1 = 1" | 
|  | foreach field $myKeyfield key $myKey { | 
|  | append req " AND $field = [makeDBFieldValue $table $field $key]" | 
|  | } | 
|  | return $req | 
|  | } | 
|  |  | 
|  | ## | 
|  | ## makekey -- Given an array containing a key-value pairs and | 
|  | # an optional  list of key fields (we use the object's keyfield | 
|  | # if none is specified)... | 
|  | # | 
|  | # if we're doing auto keys, create and return a new key, | 
|  | # otherwise if it's a single key, just return its value | 
|  | # from the array, else if it's multiple keys, return all their | 
|  | # values as a list | 
|  | ## | 
|  | method makekey {arrayName {myKeyfield ""}} { | 
|  | if {[::rivet::lempty $myKeyfield]} { set myKeyfield $keyfield } | 
|  | if {[::rivet::lempty $myKeyfield]} { | 
|  | return -code error "No -keyfield specified in object" | 
|  | } | 
|  | upvar 1 $arrayName array | 
|  |  | 
|  | ## If we're not using multiple keyfields, we want to check and see | 
|  | ## if we're using auto keys.  If we are, create a new key and | 
|  | ## return it.  If not, just return the value of the single keyfield | 
|  | ## in the array. | 
|  | if {[llength $myKeyfield] < 2} { | 
|  | if {$autokey} { | 
|  | set array($myKeyfield) [$this nextkey] | 
|  | } else { | 
|  | if {![info exists array($myKeyfield)]} { | 
|  | return -code error \ | 
|  | "${arrayName}($myKeyfield) does not exist" | 
|  | } | 
|  | } | 
|  | return $array($myKeyfield) | 
|  | } | 
|  |  | 
|  | ## We're using multiple keys.  Return a list of all the keyfield | 
|  | ## values. | 
|  | foreach field $myKeyfield { | 
|  | if {![info exists array($field)]} { | 
|  | return -code error "$field does not exist in $arrayName" | 
|  | } | 
|  | lappend key $array($field) | 
|  | } | 
|  | return $key | 
|  | } | 
|  |  | 
|  | method destroy {} { | 
|  | ::itcl::delete object $this | 
|  | } | 
|  |  | 
|  | # | 
|  | # string - execute a SQL request and only return a string of one row. | 
|  | # | 
|  | method string {req} { | 
|  | set res [exec $req] | 
|  | $res next -list val | 
|  | $res destroy | 
|  | return $val | 
|  | } | 
|  |  | 
|  | # | 
|  | # list - execute a request and return a list of the first element of each | 
|  | # row returned. | 
|  | # | 
|  | method list {req} { | 
|  | set res [exec $req] | 
|  | set list "" | 
|  | $res forall -list line { | 
|  | lappend list [lindex $line 0] | 
|  | } | 
|  | $res destroy | 
|  | return $list | 
|  | } | 
|  |  | 
|  | # | 
|  | # array - execute a request and setup an array containing elements | 
|  | # with the field names as the keys and the first row results as | 
|  | # the values | 
|  | # | 
|  | method array {req arrayName} { | 
|  | upvar 1 $arrayName $arrayName | 
|  | set res [exec $req] | 
|  | set ret [$res next -array $arrayName] | 
|  | $res destroy | 
|  | return $ret | 
|  | } | 
|  |  | 
|  | # | 
|  | # forall - execute a SQL select and iteratively fill the named array | 
|  | # with elements named with the matching field names, containing the | 
|  | # matching values, executing the specified code body for each, in turn. | 
|  | # | 
|  | method forall {req arrayName body} { | 
|  | upvar 1 $arrayName $arrayName | 
|  |  | 
|  | set res [exec $req] | 
|  |  | 
|  | $res forall -array $arrayName { | 
|  | uplevel 1 $body | 
|  | } | 
|  |  | 
|  | if {[$res error]} { | 
|  | set errinf [$res errorinfo] | 
|  | $res destroy | 
|  | return -code error "Got '$errinf' executing '$req'" | 
|  | } | 
|  |  | 
|  | set ret [$res numrows] | 
|  | $res destroy | 
|  | return $ret | 
|  | } | 
|  |  | 
|  | # | 
|  | # table_check - internal method to populate the data array with | 
|  | # a -table element containing the table name, a -keyfield element | 
|  | # containing the key field or list of key fields, and a list of | 
|  | # key-value pairs to get set into the data table. | 
|  | # | 
|  | # afterwards, it's an error if -table or -keyfield hasn't somehow been | 
|  | # determined. | 
|  | # | 
|  | protected method table_check {list {tableVar myTable} {keyVar myKeyfield}} { | 
|  | upvar 1 $tableVar $tableVar $keyVar $keyVar | 
|  | set data(-table) $table | 
|  | set data(-keyfield) $keyfield | 
|  | ::array set data $list | 
|  |  | 
|  | if {[::rivet::lempty $data(-table)]} { | 
|  | return -code error -errorcode missing_table "-table not specified in DIO object" | 
|  | } | 
|  | set $tableVar $data(-table) | 
|  |  | 
|  | if {[::rivet::lempty $data(-keyfield)]} { | 
|  | return -code error -errorcode missing_keyfield "-keyfield not specified in DIO object" | 
|  | } | 
|  |  | 
|  | set $keyVar $data(-keyfield) | 
|  | } | 
|  |  | 
|  | # | 
|  | # key_check - given a list of key fields and a list of keys, it's | 
|  | # an error if there aren't the same number of each, and if it's | 
|  | # autokey, there can't be more than one key. | 
|  | # | 
|  | protected method key_check {myKeyfield myKey} { | 
|  | if {[llength $myKeyfield] < 2} { return } | 
|  | if {$autokey} { | 
|  | return -code error "Cannot have autokey and multiple keyfields" | 
|  | } | 
|  | if {[llength $myKeyfield] != [llength $myKey]} { | 
|  | return -code error "Bad key length." | 
|  | } | 
|  | } | 
|  |  | 
|  | # | 
|  | # fetch - given a key (or list of keys) an array name, and some | 
|  | # extra key-value arguments like -table and -keyfield, fetch | 
|  | # the key into the array | 
|  | # | 
|  | method fetch {key arrayName args} { | 
|  | table_check $args | 
|  | key_check $myKeyfield $key | 
|  | upvar 1 $arrayName $arrayName | 
|  | set req "select * from $myTable" | 
|  | append req [build_key_where_clause $myKeyfield $key] | 
|  |  | 
|  | set res [$this exec $req] | 
|  | if {[$res error]} { | 
|  | set errinf [$res errorinfo] | 
|  | $res destroy | 
|  | return -code error "Got '$errinf' executing '$req'" | 
|  | } | 
|  | set rows_found [expr [$res numrows] > 0] | 
|  | $res next -array $arrayName | 
|  | $res destroy | 
|  |  | 
|  | return $rows_found | 
|  | } | 
|  |  | 
|  | # | 
|  | # store - given an array containing key-value pairs and optional | 
|  | # arguments like -table and -keyfield, insert or update the | 
|  | # corresponding table entry. | 
|  | # | 
|  | method store {arrayName args} { | 
|  | table_check $args | 
|  | upvar 1 $arrayName $arrayName $arrayName array | 
|  | if {[llength $myKeyfield] > 1 && $autokey} { | 
|  | return -code error "Cannot have autokey and multiple keyfields" | 
|  | } | 
|  |  | 
|  | set key [makekey $arrayName $myKeyfield] | 
|  | set req "select * from $myTable" | 
|  | append req [build_key_where_clause $myKeyfield $key] | 
|  | set res [exec $req] | 
|  | if {[$res error]} { | 
|  | set errinf [$res errorinfo] | 
|  | $res destroy | 
|  | return -code error "Got '$errinf' executing '$req'" | 
|  | } | 
|  | set numrows [$res numrows] | 
|  | $res destroy | 
|  |  | 
|  | if {$numrows} { | 
|  | $this update $arrayName {*}$args | 
|  | } else { | 
|  | $this insert $myTable $arrayName | 
|  | } | 
|  | return 1 | 
|  | } | 
|  |  | 
|  | # | 
|  | # update - a pure update, without store's somewhat clumsy | 
|  | # efforts to see if it needs to be an update rather than | 
|  | # an insert | 
|  | # | 
|  | method update {arrayName args} { | 
|  | table_check $args | 
|  | upvar 1 $arrayName $arrayName $arrayName row_a | 
|  |  | 
|  | set key [makekey $arrayName $myKeyfield] | 
|  |  | 
|  | set fields [::array names row_a] | 
|  | set req [build_update_query array $fields $myTable] | 
|  | append req [build_key_where_clause $myKeyfield $key] | 
|  |  | 
|  | set res [exec $req] | 
|  | if {[$res error]} { | 
|  | set errinf [$res errorinfo] | 
|  | $res destroy | 
|  | return -code error "Got '$errinf' executing '$req'" | 
|  | } | 
|  |  | 
|  | # this doesn't work on postgres, you've got to use cmdRows, | 
|  | # we need to figure out what to do with this | 
|  | set numrows [$res numrows] | 
|  | $res destroy | 
|  | return $numrows | 
|  | } | 
|  |  | 
|  | # | 
|  | # update_with_explicit_key - an update where the key is specified | 
|  | # as an argument to the proc rather than being dug out of the array | 
|  | # | 
|  | # this is a kludge until we come up with a better way to | 
|  | # solve the problem of updating a row where we actually | 
|  | # want to change the value of a key field | 
|  | # | 
|  | method update_with_explicit_key {key arrayName args} { | 
|  | table_check $args | 
|  | key_check $myKeyfield $key | 
|  | upvar 1 $arrayName $arrayName $arrayName array | 
|  |  | 
|  | set fields [::array names array] | 
|  | set req [build_update_query array $fields $myTable] | 
|  | append req [build_key_where_clause $myKeyfield $key] | 
|  |  | 
|  | set res [exec $req] | 
|  | if {[$res error]} { | 
|  | set errinf [$res errorinfo] | 
|  | $res destroy | 
|  | return -code error "Got '$errinf' executing '$req'" | 
|  | } | 
|  |  | 
|  | # this doesn't work on postgres, you've got to use cmdRows, | 
|  | # we need to figure out what to do with this | 
|  | set numrows [$res numrows] | 
|  | $res destroy | 
|  | return $numrows | 
|  | } | 
|  |  | 
|  | # | 
|  | # insert - a pure insert, without store's somewhat clumsy | 
|  | # efforts to see if it needs to be an update rather than | 
|  | # an insert -- this shouldn't require fields, it's broken | 
|  | # | 
|  | method insert {table arrayName} { | 
|  | upvar 1 $arrayName $arrayName $arrayName array | 
|  | set req [build_insert_query array [::array names array] $table] | 
|  |  | 
|  | set res [exec $req] | 
|  | if {[$res error]} { | 
|  | set errinf [$res errorinfo] | 
|  | $res destroy | 
|  | return -code error "Got '$errinf' executing '$req'" | 
|  | } | 
|  | $res destroy | 
|  | return 1 | 
|  | } | 
|  |  | 
|  | # | 
|  | # delete - delete matching record from the specified table | 
|  | # | 
|  | method delete {key args} { | 
|  | table_check $args | 
|  | set req "DELETE FROM $myTable" | 
|  | append req [build_key_where_clause $myKeyfield $key] | 
|  |  | 
|  | set res [exec $req] | 
|  | if {[$res error]} { | 
|  | set errinf [$res errorinfo] | 
|  | $res destroy | 
|  | return -code error "Got '$errinf' executing '$req'" | 
|  | } | 
|  |  | 
|  | set n_deleted_rows [$res numrows] | 
|  | $res destroy | 
|  | return $n_deleted_rows | 
|  | } | 
|  |  | 
|  | # | 
|  | # keys - return all keys in a table | 
|  | # | 
|  | method keys {args} { | 
|  | table_check $args | 
|  | set req "select * from $myTable" | 
|  | set obj [$this exec $req] | 
|  |  | 
|  | set keys "" | 
|  | $obj forall -array a { | 
|  | lappend keys [makekey a $myKeyfield] | 
|  | } | 
|  | $obj destroy | 
|  |  | 
|  | return $keys | 
|  | } | 
|  |  | 
|  | # | 
|  | # search - construct and execute a SQL select statement using | 
|  | # build_select_query style and return the result handle. | 
|  | # | 
|  | method search {args} { | 
|  | set req [eval build_select_query $args] | 
|  | return [exec $req] | 
|  | } | 
|  |  | 
|  | # | 
|  | # count - return a count of the specified (or current) table. | 
|  | # | 
|  | method count {args} { | 
|  |  | 
|  | # table_check returns an error if either a keyfield or a table were not set. | 
|  | # In order to count the rows in a table we don't need a keyfield, so we check | 
|  | # if table_check is returning missing_table and in case we rethrow the error, | 
|  | # otherwise we continue | 
|  |  | 
|  | if {[catch {table_check $args} e]} { | 
|  | if {$e != "missing_keyfield"} { | 
|  | return -code error -errorcode $e "error in table_check ($e)" | 
|  | } | 
|  | } | 
|  |  | 
|  | return [string "select count(*) from $myTable"] | 
|  | } | 
|  |  | 
|  | public method create_field_formatter {} { | 
|  | set special_fields_formatter [::DIO::formatters::[::string totitle $interface] ::DIO::formatters::#auto] | 
|  | } | 
|  |  | 
|  | protected method set_field_formatter {formatter_class} { | 
|  | if {$special_fields_formatter != ""} { $special_fields_formatter destroy } | 
|  | set special_fields_formatter [$formatter_class ::DIO::formatters::#auto] | 
|  | } | 
|  |  | 
|  | public method build_special_field {table_name field_name val {convert_to {}}} { | 
|  | return [$special_fields_formatter build $table_name $field_name $val $convert_to] | 
|  | } | 
|  |  | 
|  | public method register_special_field {table_name field_name type} { | 
|  | $special_fields_formatter register $table_name $field_name $type | 
|  | } | 
|  |  | 
|  | # method kept for compatibility | 
|  |  | 
|  | public method registerSpecialField {table_name field_name type} { | 
|  | $this register_special_field $table_name $field_name $type | 
|  | } | 
|  |  | 
|  | public method makeDBFieldValue {table_name field_name val {convert_to {}}} { | 
|  | return [$this build_special_field $table_name $field_name $val $convert_to] | 
|  | } | 
|  |  | 
|  | ## | 
|  | ## These are methods which should be defined by each individual database | 
|  | ## interface class. | 
|  | ## | 
|  | method open    {args} {} | 
|  | method close   {args} {} | 
|  | method exec    {args} {} | 
|  | method nextkey {args} {} | 
|  | method lastkey {args} {} | 
|  | method now     {}     {} | 
|  | method last_inserted_rec {} {} | 
|  | ## | 
|  | ## Functions to get and set public variables. | 
|  | ## | 
|  | method interface {{string ""}} { return [configure_variable interface $string] } | 
|  | method errorinfo {{string ""}} { return [configure_variable errorinfo $string] } | 
|  | method db {{string ""}} { return [configure_variable db $string] } | 
|  | method table {{string ""}} { return [configure_variable table $string] } | 
|  | method keyfield {{string ""}} { return [configure_variable keyfield $string] } | 
|  | method autokey {{string ""}} { return [configure_variable autokey $string] } | 
|  | method sequence {{string ""}} { return [configure_variable sequence $string] } | 
|  | method user {{string ""}} { return [configure_variable user $string] } | 
|  | method pass {{string ""}} { return [configure_variable pass $string] } | 
|  | method host {{string ""}} { return [configure_variable host $string] } | 
|  | method port {{string ""}} { return [configure_variable port $string] } | 
|  |  | 
|  | public variable special_fields_formatter "" | 
|  |  | 
|  | public variable interface   "" | 
|  | public variable errorinfo   "" | 
|  |  | 
|  | public variable db          "" | 
|  | public variable table       "" | 
|  | public variable sequence    "" | 
|  |  | 
|  | public variable user        "" | 
|  | public variable pass        "" | 
|  | public variable host        "" | 
|  | public variable port        "" | 
|  |  | 
|  | protected method handle_client_arguments {cargs} { } | 
|  |  | 
|  | public variable clientargs  "" { | 
|  | handle_client_arguments $clientargs | 
|  | } | 
|  |  | 
|  | public variable keyfield    "" { | 
|  | if {[llength $keyfield] > 1 && $autokey} { | 
|  | return -code error "Cannot have autokey and multiple keyfields" | 
|  | } | 
|  | } | 
|  |  | 
|  | public variable autokey     0 { | 
|  | if {[llength $keyfield] > 1 && $autokey} { | 
|  | return -code error "Cannot have autokey and multiple keyfields" | 
|  | } | 
|  | } | 
|  |  | 
|  | } ; ## ::itcl::class Database | 
|  |  | 
|  | # | 
|  | # DIO Result object | 
|  | # | 
|  | ::itcl::class Result { | 
|  | constructor {args} { | 
|  | eval configure $args | 
|  | } | 
|  |  | 
|  | destructor { } | 
|  |  | 
|  | method destroy {} { | 
|  | ::itcl::delete object $this | 
|  | } | 
|  |  | 
|  | # | 
|  | # configure_variable - given a variable name and a string, if the | 
|  | # string is empty return the variable name, otherwise set the | 
|  | # variable to the string. | 
|  | # | 
|  | protected method configure_variable {varName string} { | 
|  | if {[::rivet::lempty $string]} { return [cget -$varName] } | 
|  | configure -$varName $string | 
|  | } | 
|  |  | 
|  | # | 
|  | # lassign_array - given a list, an array name, and a variable number | 
|  | # of arguments consisting of variable names, assign each element in | 
|  | # the list, in turn, to elements corresponding to the variable | 
|  | # arguments, into the named array.  From TclX. | 
|  | # | 
|  | protected method lassign_array {list arrayName args} { | 
|  | upvar 1 $arrayName array | 
|  | foreach elem $list field $args { | 
|  | set array($field) $elem | 
|  | } | 
|  | } | 
|  |  | 
|  | # | 
|  | # seek - set the current row ID (our internal row cursor, if you will) | 
|  | # to the specified row ID | 
|  | # | 
|  | method seek {newrowid} { | 
|  | set rowid $newrowid | 
|  | } | 
|  |  | 
|  | method cache {{size "all"}} { | 
|  | set cacheSize $size | 
|  | if {$size == "all"} { set cacheSize $numrows } | 
|  |  | 
|  | ## Delete the previous cache array. | 
|  | catch {unset cacheArray} | 
|  |  | 
|  | set autostatus $autocache | 
|  | set currrow    $rowid | 
|  | set autocache 1 | 
|  | seek 0 | 
|  | set i 0 | 
|  | while {[next -list list]} { | 
|  | if {[incr i] >= $cacheSize} { break } | 
|  | } | 
|  | set autocache $autostatus | 
|  | seek $currrow | 
|  | set cached 1 | 
|  | } | 
|  |  | 
|  | # | 
|  | # forall -- walk the result object, executing the code body over it | 
|  | # | 
|  | method forall {type varName body} { | 
|  | upvar 1 $varName $varName | 
|  | set currrow $rowid | 
|  | seek 0 | 
|  | while {[next $type $varName]} { | 
|  | uplevel 1 $body | 
|  | } | 
|  | set rowid $currrow | 
|  | return | 
|  | } | 
|  |  | 
|  | method next {type {varName ""}} { | 
|  | set return 1 | 
|  | if {![::rivet::lempty $varName]} { | 
|  | upvar 1 $varName var | 
|  | set return 0 | 
|  | } | 
|  |  | 
|  | catch {unset var} | 
|  |  | 
|  | set list "" | 
|  | ## If we have a cached result for this row, use it. | 
|  | if {[info exists cacheArray($rowid)]} { | 
|  | set list $cacheArray($rowid) | 
|  | } else { | 
|  | set list [$this nextrow] | 
|  | if {[::rivet::lempty $list]} { | 
|  | if {$return} { return } | 
|  | set var "" | 
|  | return 0 | 
|  | } | 
|  | if {$autocache} { set cacheArray($rowid) $list } | 
|  | } | 
|  |  | 
|  | incr rowid | 
|  |  | 
|  | switch -- $type { | 
|  | "-list" { | 
|  | if {$return} { | 
|  | return $list | 
|  | } else { | 
|  | set var $list | 
|  | } | 
|  | } | 
|  | "-array" { | 
|  | if {$return} { | 
|  | foreach field $fields elem $list { | 
|  | lappend var $field $elem | 
|  | } | 
|  | return $var | 
|  | } else { | 
|  | eval lassign_array [list $list] var $fields | 
|  | } | 
|  | } | 
|  | "-keyvalue" { | 
|  | foreach field $fields elem $list { | 
|  | lappend var -$field $elem | 
|  | } | 
|  | if {$return} { return $var } | 
|  | } | 
|  | "-dict" { | 
|  | foreach field $fields elem $list { | 
|  | lappend var $field $elem | 
|  | } | 
|  | if {$return} { return [dict create {*}$var] } | 
|  | } | 
|  | default { | 
|  | incr rowid -1 | 
|  | return -code error \ | 
|  | "In-valid type: must be -list, -array, -dict or -keyvalue" | 
|  | } | 
|  | } | 
|  | return [expr [::rivet::lempty $list] == 0] | 
|  | } | 
|  |  | 
|  | public method mkdict {key} { | 
|  |  | 
|  | set query_res_d [dict create] | 
|  |  | 
|  | # we check on the first row a value | 
|  | # for the corresponding key exists | 
|  |  | 
|  | if {[$this next -dict d] > 0} { | 
|  | if {[dict exists $d $key]} { | 
|  |  | 
|  | set keyvalue [dict get $d $key] | 
|  | dict unset d $key | 
|  |  | 
|  | dict set query_res_d $keyvalue $d | 
|  | } else { | 
|  |  | 
|  | return -code error \ | 
|  | "Value for key '$key' not existing in the query results" | 
|  |  | 
|  | } | 
|  | } | 
|  |  | 
|  | # then we proceed with the remaining rows to | 
|  | # be processed | 
|  |  | 
|  | while {[$this next -dict d]} { | 
|  |  | 
|  | set keyvalue [dict get $d $key] | 
|  | dict unset d $key | 
|  |  | 
|  | dict set query_res_d $keyvalue $d | 
|  | } | 
|  |  | 
|  | return $query_res_d | 
|  | } | 
|  |  | 
|  |  | 
|  | method resultid {{string ""}} { return [configure_variable resultid $string] } | 
|  | method fields {{string ""}} { return [configure_variable fields $string] } | 
|  | method rowid {{string ""}} { return [configure_variable rowid $string] } | 
|  | method numrows {{string ""}} { return [configure_variable numrows $string] } | 
|  | method error {{string ""}} { return [configure_variable error $string] } | 
|  | method errorcode {{string ""}} { return [configure_variable errorcode $string] } | 
|  | method errorinfo {{string ""}} { return [configure_variable errorinfo $string] } | 
|  | method autocache {{string ""}} { return [configure_variable autocache $string] } | 
|  |  | 
|  | public variable resultid    "" | 
|  | public variable fields      "" | 
|  | public variable rowid       0 | 
|  | public variable numrows     0 | 
|  | public variable error       0 | 
|  | public variable errorcode   0 | 
|  | public variable errorinfo   "" | 
|  | public variable autocache   1 | 
|  |  | 
|  | protected variable cached           0 | 
|  | protected variable cacheSize        0 | 
|  | protected variable cacheArray | 
|  |  | 
|  | } ; ## ::itcl::class Result | 
|  |  | 
|  | } ; ## namespace eval DIO | 
|  |  | 
|  | package provide DIO 1.2.3 |