* contrib/gtwvg/class.prg
* contrib/gtwvg/paint.prg
* contrib/gtwvg/parthdlr.prg
* contrib/gtwvg/statbar.prg
* contrib/gtwvg/tests/_dyndlgs.prg
* contrib/gtwvg/tests/_modal.prg
* contrib/gtwvg/tests/_tbrowse.prg
* contrib/gtwvg/tests/_utils.prg
* contrib/gtwvg/tests/_wvtcls.prg
* contrib/gtwvg/tests/_xbp.prg
* contrib/gtwvg/tests/demowvg.prg
* contrib/gtwvg/tests/demowvg1.prg
* contrib/gtwvg/tests/demoxbp.prg
* contrib/gtwvg/toolbar.prg
* contrib/gtwvg/wnd.prg
* contrib/hbamf/issues.txt
* contrib/hbamf/readme.txt
* contrib/hbblink/hbblink.ch
* contrib/hbct/ct.ch
* contrib/hbct/ctcom.ch
* contrib/hbct/ctrand.prg
* contrib/hbct/cttime.prg
* contrib/hbct/doc/en/addascii.txt
* contrib/hbct/doc/en/ctmath2.txt
* contrib/hbct/doc/en/ctstr.txt
* contrib/hbct/doc/en/dattime2.txt
* contrib/hbct/doc/en/finan.txt
* contrib/hbct/doc/en/ftoc.txt
* contrib/hbct/doc/en/misc1.txt
* contrib/hbct/doc/en/trig.txt
* contrib/hbct/fcopy.prg
* contrib/hbct/getinfo.prg
* contrib/hbct/getinput.prg
* contrib/hbct/getsecrt.prg
* contrib/hbct/keysave.prg
* contrib/hbct/keysec.prg
* contrib/hbct/keytime.prg
* contrib/hbct/readme.txt
* contrib/hbct/screen3.prg
* contrib/hbct/scrmark.prg
* contrib/hbct/showtime.prg
* contrib/hbct/tempfile.prg
* contrib/hbformat/utils/hbformat.prg
* contrib/hbfship/isdb.prg
* contrib/hbgd/gdimage.prg
* contrib/hbgd/tests/gdtestcl.prg
* contrib/hbgd/tests/test_out.prg
* contrib/hbgd/tests/tostring.prg
* contrib/hbgt/doc/en/hbgt.txt
* contrib/hbgt/tests/test.prg
* contrib/hbhpdf/tests/harupdf.prg
* contrib/hbhttpd/tests/carts.dbf
* contrib/hbhttpd/tests/items.dbf
* contrib/hbhttpd/tests/users.dbf
* contrib/hbmisc/doc/en/dates2.txt
* contrib/hbmisc/fcomma.prg
* contrib/hbmisc/stringp.prg
* contrib/hbmisc/tests/testhbf.prg
* contrib/hbmisc/udpds.prg
* contrib/hbmxml/tests/custom.prg
* contrib/hbmysql/diff-en.txt
* contrib/hbmysql/diff-es.txt
* contrib/hbmysql/tmysql.prg
* contrib/hbnetio/netiomt.prg
* contrib/hbnetio/utils/hbnetio/netiomgm.hb
* contrib/hbnf/adapter.prg
* contrib/hbnf/aredit.prg
* contrib/hbnf/clrsel.prg
* contrib/hbnf/default.prg
* contrib/hbnf/dispmsg.prg
* contrib/hbnf/floptst.prg
* contrib/hbnf/ftint86.ch
* contrib/hbnf/ftmenuto.ch
* contrib/hbnf/lastday.prg
* contrib/hbnf/linked.prg
* contrib/hbnf/menuto.prg
* contrib/hbnf/metaph.prg
* contrib/hbnf/nwsem.prg
* contrib/hbnf/ontick.prg
* contrib/hbnf/pegs.prg
* contrib/hbnf/popadder.prg
* contrib/hbnf/savesets.prg
* contrib/hbnf/sinkey.prg
* contrib/hbnf/tbwhile.prg
* contrib/hbnf/tempfile.prg
* contrib/hbnf/vidcur.prg
* contrib/hbnf/video2.prg
* contrib/hbnf/woy.prg
* contrib/hbodbc/tests/testodbc.prg
* contrib/hbodbc/todbc.prg
* contrib/hbpgsql/postgres.ch
* contrib/hbtip/encoder.prg
* contrib/hbtip/thtml.prg
* contrib/hbtpathy/tests/testtp.prg
* contrib/hbwin/tests/dlg.prg
* contrib/hbwin/tests/olesrv1.prg
* contrib/hbwin/tests/olesrv3.prg
* contrib/hbwin/tests/olesrv4.prg
* contrib/hbwin/tests/oletst4.prg
* contrib/hbwin/tests/testprn.prg
* contrib/hbwin/win_tbmp.prg
* contrib/hbxpp/browdbx.prg
* contrib/hbxpp/runshell.prg
* contrib/hbxpp/tthreadx.prg
* contrib/hbxpp/typefilx.prg
* contrib/hbxpp/xppop.prg
* contrib/rddads/doc/en/adsfuncs.txt
* contrib/rddads/doc/en/readme.txt
* contrib/rddads/tests/datad.prg
* contrib/rddsql/readme.txt
* contrib/xhb/cstruct.prg
* contrib/xhb/dirrec.prg
* contrib/xhb/diskhb.prg
* contrib/xhb/hbcompat.ch
* contrib/xhb/hbcomprs.prg
* contrib/xhb/hterrsys.prg
* contrib/xhb/htjlist.prg
* contrib/xhb/htmutil.prg
* contrib/xhb/stream.prg
* contrib/xhb/tcgi.prg
* contrib/xhb/thtm.prg
* contrib/xhb/ttable.prg
* contrib/xhb/txml.prg
* contrib/xhb/xcstr.prg
* contrib/xhb/xhberr.prg
* contrib/xhb/xhbmemo.prg
* contrib/xhb/xhbtedit.prg
* contrib/xhb/xhbver.prg
* doc/cmpopt.txt
* extras/gfspell/spell.ng
* extras/gfspell/spell.prg
* extras/gfspell/words.dbf
* extras/gtwvw/docs/funclist.txt
* extras/gtwvw/docs/whatsnew.txt
* extras/gtwvw/tests/_wvwmous.prg
* extras/gtwvw/tests/cbtest6.prg
* extras/gtwvw/tests/drawimg.prg
* extras/gtwvw/tests/ebtest7.prg
* extras/gtwvw/tests/inpfocus.prg
* extras/gtwvw/tests/maximize.prg
* extras/gtwvw/tests/prog0.prg
* extras/gtwvw/tests/prog1.prg
* extras/gtwvw/tests/prog2.prg
* extras/gtwvw/tests/wvwtest9.prg
* extras/gtwvw/wvt2wvw.ch
* extras/guestbk/guestbk.prg
* extras/hbdoc/tmplates.prg
* extras/hbvpdf/core.prg
* extras/httpsrv/cgifunc.prg
* extras/httpsrv/cookie.prg
* extras/httpsrv/session.prg
* extras/httpsrv/uhttpd.prg
* extras/rddado/adordd.ch
* extras/rddado/adordd.prg
* include/set.ch
* src/debug/debugger.prg
* src/rdd/dbfuncs.prg
* src/rdd/dbjoin.prg
* src/rdd/dblist.prg
* src/rdd/dbsort.prg
* src/rdd/dbstrux.prg
* src/rdd/dbstruxu.prg
* src/rdd/dbtotal.prg
* src/rdd/dbupdat.prg
* src/rdd/hbsix/sxcompat.prg
* src/rdd/hbsix/sxini.prg
* src/rdd/hbsix/sxtrig.prg
* src/rdd/usrrdd/rdds/arrayrdd.prg
* src/rdd/usrrdd/rdds/logrdd.prg
* src/rdd/usrrdd/rdds/rlcdx.prg
* src/rtl/achoice.prg
* src/rtl/adir.prg
* src/rtl/alert.prg
* src/rtl/altd.prg
* src/rtl/browdb.prg
* src/rtl/dbedit.prg
* src/rtl/devoutp.prg
* src/rtl/dircmd.prg
* src/rtl/hbi18n2.prg
* src/rtl/input.prg
* src/rtl/libname.prg
* src/rtl/listbox.prg
* src/rtl/memvarbl.prg
* src/rtl/radiobhb.prg
* src/rtl/readkey.prg
* src/rtl/readvar.prg
* src/rtl/setfunc.prg
* src/rtl/setta.prg
* src/rtl/tbrowse.prg
* src/rtl/teditor.prg
* src/rtl/text.prg
* src/rtl/tget.prg
* src/rtl/tgetlist.prg
* src/rtl/tobject.prg
* src/rtl/typefile.prg
* src/rtl/valtoexp.prg
* src/rtl/wait.prg
* src/vm/harbinit.prg
* tests/ac_test.prg
* tests/ac_test2.prg
* tests/adirtest.prg
* tests/aliaslck.prg
* tests/bldtest/bldtest.c
* tests/clsnv.prg
* tests/codebl.prg
* tests/debugtst.prg
* tests/devtest.prg
* tests/disptest.prg
* tests/gtcolors.prg
* tests/gtkeys.prg
* tests/gtwin.prg
* tests/gtxfnt.prg
* tests/inkeytst.prg
* tests/linecont.prg
* tests/mathtest.prg
* tests/memvar.prg
* tests/newrdd.prg
* tests/output.prg
* tests/readhrb.prg
* tests/seconds.prg
* tests/set_test.prg
* tests/tb1.prg
* tests/testdyn.prg
* tests/testdyn1.prg
* tests/testhrb.prg
* tests/testhtml.prg
* tests/testidle.prg
* tests/testpre.prg
* tests/testrdd2.prg
* tests/testsha1.prg
* tests/teststr.prg
* tests/ticktime.prg
* tests/transtst.prg
* tests/tstcolor.prg
* tests/tstdbi.prg
* tests/tstmacro.prg
* tests/uc16_gen.prg
* tests/usrrdd/exarr.prg
* tests/usrrdd/exlog.prg
* tests/utf8at.prg
* tests/version.prg
* tests/videotst.prg
* tests/vidtest.prg
* tests/wvtext.prg
* utils/hbmk2/hbmk2.prg
* utils/hbtest/hbtest.prg
* utils/hbtest/rt_class.prg
* utils/hbtest/rt_date.prg
* utils/hbtest/rt_hvm.prg
* utils/hbtest/rt_hvma.prg
* utils/hbtest/rt_math.prg
* utils/hbtest/rt_misc.prg
* utils/hbtest/rt_mt.prg
* utils/hbtest/rt_str.prg
* utils/hbtest/rt_stra.prg
* utils/hbtest/rt_trans.prg
* run case fixer script on all non-C file
except some certain files, which I routinely
skip when formatting.
* minor manual cleanups along the way
134 lines
5.4 KiB
Plaintext
134 lines
5.4 KiB
Plaintext
/*
|
|
* $Id$
|
|
*/
|
|
|
|
Simple SQL Interface for Harbour
|
|
|
|
|
|
|
|
1. Introduction
|
|
|
|
Simple SQL interface implements accessing SQL query result via RDD
|
|
interface. It is not intended to be replacement for "transparent" move of
|
|
DBFCDX application to SQL world.
|
|
|
|
I want to discuss this in more detail. Many current RDDs for SQL servers
|
|
(ex. SQLRDD from xHarbour.com) tries to make a feeling you are working with
|
|
DBF file, but not with SQL database. SQL server does not support many
|
|
features, ex. RecNo(), deleted flag, file locks, record locks. These RDDs
|
|
are emulating these features to make feeling of DBF. Deleted() function is
|
|
emulated by creating additional table columns to store delete flag. Some
|
|
"hidden system" tables are used to register locking operations and emulate
|
|
record and file locks in DBF style. The idea of SQL query is also lost. If
|
|
you do a simple loop
|
|
|
|
dbUseArea(, "select * from my_table" )
|
|
DO WHILE ! Eof()
|
|
somefunc( FIELD->some_sql_field )
|
|
dbSkip()
|
|
ENDDO
|
|
|
|
RDD usualy will read SQL rows in portions, let's say 100 records per query.
|
|
So, hidden queries are generated. If you are using indexes these queries
|
|
are really complicated. Let's have index on FIELD1 + Str(FIELD2). A seek to
|
|
value cValue1 + Str(nValue2) will generate a query like:
|
|
|
|
SELECT * FROM my_table
|
|
WHERE (FIELD1 == cValue1 and FIELD2 >= nValue2) or FIELD1 > cValue1
|
|
ORDER BY FIELD1, FIELD2, _RECNO
|
|
LIMIT 100
|
|
|
|
After evaluation of first 100 cached records, next query will be generated:
|
|
|
|
SELECT * FROM my_table
|
|
WHERE (FIELD1 == cLastField1 and FIELD2 == nLastValue2 and _RECNO > nLastRecno) or
|
|
(FIELD1 == cLastField1 and FIELD2 > nLastValue2) or
|
|
FIELD1 > cLastValue1
|
|
ORDER BY FIELD1, FIELD2, _RECNO
|
|
LIMIT 100
|
|
|
|
To optimize these queries the SQL index expresion should be
|
|
"FIELD1,FIELD2,_RECNO", but not "FIELD1,FIELD2" as written in INDEX ON
|
|
command.
|
|
|
|
"Simple SQL interface" is too long to repeat every time I want to
|
|
address this library. I'll also use acronym "SSI" to address it.
|
|
|
|
The idea of SSI is different. It does not make hidden queries. All
|
|
queries should be made explicitly by programmer. SSI gives access to query
|
|
result via RDD interface, it does not tries to emulate DBF and be
|
|
"plug-and-play" solution for DBF to SQL migration. If you do
|
|
|
|
dbUseArea(, "select * from my_table")
|
|
|
|
all query (it could contain millions of records!) will be cached.
|
|
|
|
The features of SSI approach are:
|
|
|
|
- It's possible to access SQL database of other applications. Other
|
|
applications usualy does not follow agreement of "plug-and-play" SQL drivers
|
|
about additional DELETED column, _RECNO in the end of index expression, etc.
|
|
Access of SQL database of other applications is sometimes not possible.
|
|
|
|
- It's query oriented. That means a simple DO WHILE ! Eof() loop will iterate
|
|
each records once and only once. This is not true for "plug-and-play" SQL
|
|
drivers, if indexing is used. Just like in the case of loop over DBF file.
|
|
It is not guaranteed that all records are included! Yes! If key value of the
|
|
first record in index is changed to be the last record in index during the
|
|
phase of record processing, DO WHILE ! Eof() loop will iterate only this
|
|
single records even if the database contains millions of records. Your sould
|
|
do FLock() on DBF to guarantee the records are not changed. Do you use FLock()
|
|
before readonly DO WHILE ! Eof() loops? :)
|
|
|
|
|
|
|
|
2. Architecture
|
|
|
|
|
|
+-------------+
|
|
| |
|
|
| SQLMIX RDD |
|
|
| |
|
|
+-------------+
|
|
| ^
|
|
V |
|
|
+-------------+ +---------+
|
|
| |--->| |
|
|
| SQLBASE RDD | | SDD |
|
|
| |<---| |
|
|
+-------------+ +---------+
|
|
|
|
|
|
SQLBASE RDD implements basic functionality for accessing SQL query result
|
|
via RDD interface. This RDD could be used, if indexing of query result is not
|
|
necessary or all indexing is done by SQL server (by using ORDER BY clause).
|
|
|
|
SQLMIX RDD implements indexing of query result. This indexing is not
|
|
related to SQL server ORDER BY clause. SQLMIX do indexing of the query on the
|
|
client side.
|
|
|
|
SDD is acronym for Sql Database Driver. RDD is used to implement access
|
|
of different database formats like DBF, SDF, etc. SDD is used to implement
|
|
access of different SQL databases. Every SQL server (MySQL, PostgreSQL, etc.)
|
|
has a corresponding SDD. SDD driver implements a specific part of data
|
|
exchange interface between SQLBASE and SQL server.
|
|
|
|
A few additional functions are also implemented, ex. HB_SQLCONNECT().
|
|
Usualy these functions are just a shorter version of corresponding rddInfo()
|
|
call.
|
|
|
|
|
|
|
|
3. Modifying database
|
|
|
|
SSI presents a query result via RDD interface and generates no hidden
|
|
SQL queries. So, how database can be changed? Does dbAppend() and FieldPut()
|
|
works, or is it readonly SQL interface?
|
|
dbAppend(), FieldPut() and other similiar functions work on cached query
|
|
result, i.e. query can be appended by new rows and field values can be
|
|
changed, but SQL database is not changed. dbCreate() function can also be
|
|
used to create an "empty query result" but no table is created on SQL server.
|
|
So, SSI can also be used as implementation of "array RDD".
|
|
The programmer must call SQL command explicitly to modify SQL tables.
|
|
SSI provides a method to detect which cached rows was changed or appended.
|