2013-02-07 13:34:27 +04:00
|
|
|
/************* TabTbl C++ Program Source Code File (.CPP) **************/
|
|
|
|
/* PROGRAM NAME: TABTBL */
|
|
|
|
/* ------------- */
|
2013-04-29 13:50:20 +02:00
|
|
|
/* Version 1.5 */
|
2013-02-07 13:34:27 +04:00
|
|
|
/* */
|
|
|
|
/* COPYRIGHT: */
|
|
|
|
/* ---------- */
|
2013-02-08 03:27:12 +01:00
|
|
|
/* (C) Copyright to PlugDB Software Development 2008-2013 */
|
2013-02-07 13:34:27 +04:00
|
|
|
/* Author: Olivier BERTRAND */
|
|
|
|
/* */
|
|
|
|
/* WHAT THIS PROGRAM DOES: */
|
|
|
|
/* ----------------------- */
|
|
|
|
/* This program are the TDBTBL class DB routines. */
|
|
|
|
/* */
|
|
|
|
/* WHAT YOU NEED TO COMPILE THIS PROGRAM: */
|
|
|
|
/* -------------------------------------- */
|
|
|
|
/* */
|
|
|
|
/* REQUIRED FILES: */
|
|
|
|
/* --------------- */
|
|
|
|
/* TABTBL.CPP - Source code */
|
|
|
|
/* PLGDBSEM.H - DB application declaration file */
|
|
|
|
/* TABDOS.H - TABDOS classes declaration file */
|
|
|
|
/* TABTBL.H - TABTBL classes declaration file */
|
|
|
|
/* GLOBAL.H - Global declaration file */
|
|
|
|
/* */
|
|
|
|
/* REQUIRED LIBRARIES: */
|
|
|
|
/* ------------------- */
|
|
|
|
/* Large model C library */
|
|
|
|
/* */
|
|
|
|
/* REQUIRED PROGRAMS: */
|
|
|
|
/* ------------------ */
|
|
|
|
/* IBM, Borland, GNU or Microsoft C++ Compiler and Linker */
|
|
|
|
/* */
|
|
|
|
/***********************************************************************/
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Include relevant section of system dependant header files. */
|
|
|
|
/***********************************************************************/
|
|
|
|
//#include "sql_base.h"
|
|
|
|
#include "my_global.h"
|
|
|
|
#if defined(WIN32)
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#if defined(__BORLANDC__)
|
|
|
|
#define __MFC_COMPAT__ // To define min/max as macro
|
|
|
|
#endif
|
|
|
|
//#include <windows.h>
|
|
|
|
#else
|
|
|
|
#if defined(UNIX)
|
|
|
|
#include <fnmatch.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "osutil.h"
|
|
|
|
#else
|
|
|
|
//#include <io.h>
|
|
|
|
#endif
|
|
|
|
//#include <fcntl.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Include application header files: */
|
|
|
|
/***********************************************************************/
|
|
|
|
#include "table.h" // MySQL table definitions
|
|
|
|
#include "global.h" // global declarations
|
|
|
|
#include "plgdbsem.h" // DB application declarations
|
|
|
|
#include "reldef.h" // DB definition declares
|
|
|
|
//#include "filter.h" // FILTER classes dcls
|
|
|
|
#include "filamtxt.h"
|
|
|
|
#include "tabcol.h"
|
|
|
|
#include "tabdos.h" // TDBDOS and DOSCOL class dcls
|
2013-04-29 13:50:20 +02:00
|
|
|
#include "tabtbl.h"
|
2013-04-12 18:30:15 +02:00
|
|
|
#if defined(MYSQL_SUPPORT)
|
|
|
|
#include "tabmysql.h"
|
|
|
|
#endif // MYSQL_SUPPORT
|
2013-02-07 13:34:27 +04:00
|
|
|
#include "ha_connect.h"
|
|
|
|
#include "mycat.h" // For GetHandler
|
|
|
|
|
|
|
|
extern "C" int trace;
|
|
|
|
|
|
|
|
/* ---------------------------- Class TBLDEF ---------------------------- */
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
/* Constructor. */
|
|
|
|
/**************************************************************************/
|
|
|
|
TBLDEF::TBLDEF(void)
|
|
|
|
{
|
2013-04-29 13:50:20 +02:00
|
|
|
//To_Tables = NULL;
|
2013-02-07 13:34:27 +04:00
|
|
|
Ntables = 0;
|
|
|
|
Pseudo = 3;
|
|
|
|
} // end of TBLDEF constructor
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
/* DefineAM: define specific AM block values from XDB file. */
|
|
|
|
/**************************************************************************/
|
|
|
|
bool TBLDEF::DefineAM(PGLOBAL g, LPCSTR am, int poff)
|
|
|
|
{
|
|
|
|
char *tablist, *dbname;
|
|
|
|
|
|
|
|
Desc = "Table list table";
|
2013-03-25 11:07:45 +01:00
|
|
|
tablist = Cat->GetStringCatInfo(g, "Tablist", "");
|
2013-04-29 13:50:20 +02:00
|
|
|
dbname = Cat->GetStringCatInfo(g, "Dbname", "*");
|
2013-02-07 13:34:27 +04:00
|
|
|
Ntables = 0;
|
|
|
|
|
|
|
|
if (*tablist) {
|
2013-04-29 13:50:20 +02:00
|
|
|
char *p, *pn, *pdb;
|
|
|
|
PTABLE tbl;
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
for (pdb = tablist; ;) {
|
|
|
|
if ((p = strchr(pdb, ',')))
|
|
|
|
*p = 0;
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
// Analyze the table name, it may have the format:
|
2013-02-07 13:34:27 +04:00
|
|
|
// [dbname.]tabname
|
|
|
|
if ((pn = strchr(pdb, '.'))) {
|
|
|
|
*pn++ = 0;
|
|
|
|
} else {
|
|
|
|
pn = pdb;
|
|
|
|
pdb = dbname;
|
|
|
|
} // endif p
|
|
|
|
|
|
|
|
// Allocate the TBLIST block for that table
|
2013-04-29 13:50:20 +02:00
|
|
|
tbl = new(g) XTAB(pn);
|
|
|
|
tbl->SetQualifier(pdb);
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
if (trace)
|
2013-04-29 13:50:20 +02:00
|
|
|
htrc("TBL: Name=%s db=%s\n", tbl->GetName(), tbl->GetQualifier());
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
// Link the blocks
|
2013-04-29 13:50:20 +02:00
|
|
|
if (Tablep)
|
|
|
|
Tablep->Link(tbl);
|
|
|
|
else
|
|
|
|
Tablep = tbl;
|
|
|
|
|
2013-02-07 13:34:27 +04:00
|
|
|
Ntables++;
|
|
|
|
|
|
|
|
if (p)
|
|
|
|
pdb = pn + strlen(pn) + 1;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
|
|
|
|
} // endfor pdb
|
|
|
|
|
2013-03-25 11:07:45 +01:00
|
|
|
Maxerr = Cat->GetIntCatInfo("Maxerr", 0);
|
|
|
|
Accept = (Cat->GetBoolCatInfo("Accept", 0) != 0);
|
2013-02-07 13:34:27 +04:00
|
|
|
} // endif fsec || tablist
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
} // end of DefineAM
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* GetTable: makes a new Table Description Block. */
|
|
|
|
/***********************************************************************/
|
|
|
|
PTDB TBLDEF::GetTable(PGLOBAL g, MODE m)
|
|
|
|
{
|
2013-04-29 13:50:20 +02:00
|
|
|
if (Catfunc == FNC_COL)
|
|
|
|
return new(g) TDBTBC(this);
|
|
|
|
else
|
|
|
|
return new(g) TDBTBL(this);
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
} // end of GetTable
|
|
|
|
|
|
|
|
/* ------------------------- Class TDBTBL ---------------------------- */
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* TDBTBL constructors. */
|
|
|
|
/***********************************************************************/
|
2013-04-29 13:50:20 +02:00
|
|
|
TDBTBL::TDBTBL(PTBLDEF tdp) : TDBPRX(tdp)
|
2013-02-07 13:34:27 +04:00
|
|
|
{
|
|
|
|
Tablist = NULL;
|
|
|
|
CurTable = NULL;
|
2013-04-29 13:50:20 +02:00
|
|
|
//Tdbp = NULL;
|
2013-02-07 13:34:27 +04:00
|
|
|
Accept = tdp->Accept;
|
|
|
|
Maxerr = tdp->Maxerr;
|
|
|
|
Nbf = 0;
|
|
|
|
Rows = 0;
|
|
|
|
Crp = 0;
|
|
|
|
// NTables = 0;
|
|
|
|
// iTable = 0;
|
|
|
|
} // end of TDBTBL standard constructor
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Allocate TBL column description block. */
|
|
|
|
/***********************************************************************/
|
|
|
|
PCOL TDBTBL::MakeCol(PGLOBAL g, PCOLDEF cdp, PCOL cprec, int n)
|
|
|
|
{
|
2013-04-29 13:50:20 +02:00
|
|
|
return new(g) PRXCOL(cdp, this, cprec, n);
|
2013-02-07 13:34:27 +04:00
|
|
|
} // end of MakeCol
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* InsertSpecialColumn: Put a special column ahead of the column list.*/
|
|
|
|
/***********************************************************************/
|
|
|
|
PCOL TDBTBL::InsertSpecialColumn(PGLOBAL g, PCOL scp)
|
|
|
|
{
|
|
|
|
PCOL colp;
|
|
|
|
|
|
|
|
if (!scp->IsSpecial())
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (scp->GetAmType() == TYPE_AM_TABID)
|
|
|
|
// This special column is handled locally
|
|
|
|
colp = new((TIDBLK*)scp) TBTBLK(scp->GetValue());
|
|
|
|
else // Other special columns are treated normally
|
|
|
|
colp = scp;
|
|
|
|
|
|
|
|
colp->SetNext(Columns);
|
|
|
|
Columns = colp;
|
|
|
|
return colp;
|
|
|
|
} // end of InsertSpecialColumn
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
#if 0
|
2013-02-07 13:34:27 +04:00
|
|
|
/***********************************************************************/
|
|
|
|
/* Get the PTDB of a table of the list. */
|
|
|
|
/***********************************************************************/
|
|
|
|
PTDB TDBTBL::GetSubTable(PGLOBAL g, PTBL tblp, PTABLE tabp)
|
|
|
|
{
|
2013-04-29 13:50:20 +02:00
|
|
|
//char *db;
|
|
|
|
bool mysql;
|
2013-02-07 13:34:27 +04:00
|
|
|
PTDB tdbp = NULL;
|
|
|
|
TABLE_SHARE *s;
|
|
|
|
PCATLG cat = To_Def->GetCat();
|
2013-04-12 18:30:15 +02:00
|
|
|
PHC hc = ((MYCAT*)cat)->GetHandler();
|
|
|
|
THD *thd = (hc->GetTable())->in_use;
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
if (!thd)
|
|
|
|
return NULL; // Should not happen anymore
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
//if (tblp->DB)
|
|
|
|
// db = tblp->DB;
|
|
|
|
//else
|
|
|
|
// db = (char*)hc->GetDBName(NULL);
|
2013-02-07 13:34:27 +04:00
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
//if (!(s = GetTableShare(g, thd, db, tblp->Name, mysql)))
|
|
|
|
if (!(s = GetTableShare(g, thd, tblp->DB, tblp->Name, mysql)))
|
|
|
|
return NULL;
|
2013-02-07 13:34:27 +04:00
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
if (mysql) {
|
2013-04-12 18:30:15 +02:00
|
|
|
#if defined(MYSQL_SUPPORT)
|
2013-04-29 13:50:20 +02:00
|
|
|
// Access sub-table via MySQL API
|
|
|
|
if (!(tdbp= cat->GetTable(g, tabp, MODE_READ, "MYSQL"))) {
|
|
|
|
sprintf(g->Message, "Cannot access %s.%s", tblp->DB, tblp->Name);
|
|
|
|
goto err;
|
|
|
|
} // endif Define
|
2013-04-12 18:30:15 +02:00
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
if (tabp->GetQualifier())
|
|
|
|
((PTDBMY)tdbp)->SetDatabase(tabp->GetQualifier());
|
2013-04-12 18:30:15 +02:00
|
|
|
|
|
|
|
#else // !MYSQL_SUPPORT
|
2013-04-29 13:50:20 +02:00
|
|
|
sprintf(g->Message, "%s.%s is not a CONNECT table",
|
|
|
|
db, tblp->Name);
|
|
|
|
goto err;
|
2013-04-12 18:30:15 +02:00
|
|
|
#endif // MYSQL_SUPPORT
|
2013-04-29 13:50:20 +02:00
|
|
|
} else {
|
|
|
|
// Sub-table is a CONNECT table
|
|
|
|
hc->tshp = s;
|
|
|
|
tdbp = cat->GetTable(g, tabp);
|
|
|
|
hc->tshp = NULL;
|
|
|
|
} // endif plugin
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
if (trace && tdbp)
|
|
|
|
htrc("Subtable %s in %s\n",
|
|
|
|
tblp->Name, SVP(((PTDBASE)tdbp)->GetDef()->GetDB()));
|
2013-04-29 13:50:20 +02:00
|
|
|
|
|
|
|
err:
|
2013-02-07 13:34:27 +04:00
|
|
|
free_table_share(s);
|
|
|
|
return tdbp;
|
|
|
|
} // end of GetSubTable
|
2013-04-29 13:50:20 +02:00
|
|
|
#endif // 0
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Initializes the table table list. */
|
|
|
|
/***********************************************************************/
|
|
|
|
bool TDBTBL::InitTableList(PGLOBAL g)
|
|
|
|
{
|
|
|
|
char *colname;
|
|
|
|
int n, colpos;
|
2013-04-29 13:50:20 +02:00
|
|
|
PTABLE tp, tabp;
|
2013-02-07 13:34:27 +04:00
|
|
|
PTDB tdbp;
|
|
|
|
PCOL colp;
|
|
|
|
PTBLDEF tdp = (PTBLDEF)To_Def;
|
|
|
|
|
|
|
|
// PlugSetPath(filename, Tdbp->GetFile(g), Tdbp->GetPath());
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
for (n = 0, tp = tdp->Tablep; tp; tp = tp->GetNext()) {
|
|
|
|
if (TestFil(g, To_Filter, tp)) {
|
|
|
|
tabp = new(g) XTAB(tp);
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
// Get the table description block of this table
|
2013-04-29 13:50:20 +02:00
|
|
|
if (!(tdbp = GetSubTable(g, tabp))) {
|
2013-02-07 13:34:27 +04:00
|
|
|
if (++Nbf > Maxerr)
|
|
|
|
return TRUE; // Error return
|
|
|
|
else
|
|
|
|
continue; // Skip this table
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
} // endif tdbp
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
// We must allocate subtable columns before GetMaxSize is called
|
|
|
|
// because some (PLG, ODBC?) need to have their columns attached.
|
|
|
|
// Real initialization will be done later.
|
|
|
|
for (PCOL cp = Columns; cp; cp = cp->GetNext())
|
|
|
|
if (!cp->IsSpecial()) {
|
|
|
|
colname = cp->GetName();
|
2013-04-29 13:50:20 +02:00
|
|
|
colpos = ((PPRXCOL)cp)->Colnum;
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
// We try first to get the column by name
|
|
|
|
if (!(colp = tdbp->ColDB(g, colname, 0)) && colpos)
|
|
|
|
// When unsuccessful, if a column number was specified
|
|
|
|
// try to get the column by its position in the table
|
|
|
|
colp = tdbp->ColDB(g, NULL, colpos);
|
|
|
|
|
|
|
|
if (!colp) {
|
|
|
|
if (!Accept) {
|
|
|
|
sprintf(g->Message, MSG(NO_MATCHING_COL),
|
|
|
|
colname, tdbp->GetName());
|
|
|
|
return TRUE; // Error return
|
|
|
|
} // endif !Accept
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
} else // this is needed by some tables (which?)
|
2013-02-07 13:34:27 +04:00
|
|
|
colp->SetColUse(cp->GetColUse());
|
|
|
|
|
|
|
|
} // endif !special
|
|
|
|
|
|
|
|
if (Tablist)
|
|
|
|
Tablist->Link(tabp);
|
|
|
|
else
|
|
|
|
Tablist = tabp;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
} // endif filp
|
|
|
|
|
|
|
|
} // endfor tblp
|
|
|
|
|
|
|
|
//NumTables = n;
|
|
|
|
To_Filter = NULL; // To avoid doing it several times
|
|
|
|
return FALSE;
|
|
|
|
} // end of InitTableList
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Test the tablename against the pseudo "local" filter. */
|
|
|
|
/***********************************************************************/
|
2013-04-29 13:50:20 +02:00
|
|
|
bool TDBTBL::TestFil(PGLOBAL g, PFIL filp, PTABLE tabp)
|
2013-02-07 13:34:27 +04:00
|
|
|
{
|
|
|
|
char *fil, op[8], tn[NAME_LEN];
|
|
|
|
bool neg;
|
|
|
|
|
|
|
|
if (!filp)
|
|
|
|
return TRUE;
|
|
|
|
else if (strstr(filp, " OR ") || strstr(filp, " AND "))
|
|
|
|
return TRUE; // Not handled yet
|
|
|
|
else
|
|
|
|
fil = filp + (*filp == '(' ? 1 : 0);
|
|
|
|
|
|
|
|
if (sscanf(fil, "TABID %s", op) != 1)
|
|
|
|
return TRUE; // ignore invalid filter
|
|
|
|
|
|
|
|
if ((neg = !strcmp(op, "NOT")))
|
|
|
|
strcpy(op, "IN");
|
|
|
|
|
|
|
|
if (!strcmp(op, "=")) {
|
|
|
|
// Temporarily, filter must be "TABID = 'value'" only
|
|
|
|
if (sscanf(fil, "TABID = '%[^']'", tn) != 1)
|
|
|
|
return TRUE; // ignore invalid filter
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
return !stricmp(tn, tabp->GetName());
|
2013-02-07 13:34:27 +04:00
|
|
|
} else if (!strcmp(op, "IN")) {
|
|
|
|
char *p, *tnl = (char*)PlugSubAlloc(g, NULL, strlen(fil) - 10);
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (neg)
|
|
|
|
n = sscanf(fil, "TABID NOT IN (%[^)])", tnl);
|
|
|
|
else
|
|
|
|
n = sscanf(fil, "TABID IN (%[^)])", tnl);
|
|
|
|
|
|
|
|
if (n != 1)
|
|
|
|
return TRUE; // ignore invalid filter
|
|
|
|
|
|
|
|
while (tnl) {
|
|
|
|
if ((p = strchr(tnl, ',')))
|
|
|
|
*p++ = 0;
|
|
|
|
|
|
|
|
if (sscanf(tnl, "'%[^']'", tn) != 1)
|
|
|
|
return TRUE; // ignore invalid filter
|
2013-04-29 13:50:20 +02:00
|
|
|
else if (!stricmp(tn, tabp->GetName()))
|
2013-02-07 13:34:27 +04:00
|
|
|
return !neg; // Found
|
|
|
|
|
|
|
|
tnl = p;
|
|
|
|
} // endwhile
|
|
|
|
|
|
|
|
return neg; // Not found
|
|
|
|
} // endif op
|
|
|
|
|
|
|
|
return TRUE; // invalid operator
|
|
|
|
} // end of TestFil
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
#if 0
|
2013-02-07 13:34:27 +04:00
|
|
|
/***********************************************************************/
|
|
|
|
/* TBL GetProgMax: get the max value for progress information. */
|
|
|
|
/***********************************************************************/
|
|
|
|
int TDBTBL::GetProgMax(PGLOBAL g)
|
|
|
|
{
|
2013-04-29 13:50:20 +02:00
|
|
|
int n, pmx = 0;
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
if (!Tablist && InitTableList(g))
|
|
|
|
return -1;
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
for (PTABLE tabp = Tablist; tabp; tblp = tabp->GetNext())
|
|
|
|
if ((n = tabp->GetTo_Tdb()->GetProgMax(g)) > 0)
|
2013-02-07 13:34:27 +04:00
|
|
|
pmx += n;
|
|
|
|
|
|
|
|
return pmx;
|
|
|
|
} // end of GetProgMax
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* TBL GetProgCur: get the current value for progress information. */
|
|
|
|
/***********************************************************************/
|
|
|
|
int TDBTBL::GetProgCur(void)
|
|
|
|
{
|
|
|
|
return Crp + Tdbp->GetProgCur();
|
|
|
|
} // end of GetProgCur
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* TBL Cardinality: returns table cardinality in number of rows. */
|
|
|
|
/* This function can be called with a null argument to test the */
|
|
|
|
/* availability of Cardinality implementation (1 yes, 0 no). */
|
|
|
|
/* Can be used on Multiple FIX table only. */
|
|
|
|
/***********************************************************************/
|
|
|
|
int TDBTBL::Cardinality(PGLOBAL g)
|
|
|
|
{
|
|
|
|
if (!g)
|
|
|
|
return Tdbp->Cardinality(g);
|
|
|
|
|
|
|
|
if (!Tablist && InitTableList(g))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
int n, card = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < NumFiles; i++) {
|
|
|
|
Tdbp->SetFile(g, Filenames[i]);
|
|
|
|
Tdbp->ResetSize();
|
|
|
|
|
|
|
|
if ((n = Tdbp->Cardinality(g)) < 0) {
|
|
|
|
// strcpy(g->Message, MSG(BAD_CARDINALITY));
|
|
|
|
return -1;
|
|
|
|
} // endif n
|
|
|
|
|
|
|
|
card += n;
|
|
|
|
} // endfor i
|
|
|
|
|
|
|
|
return card;
|
|
|
|
} // end of Cardinality
|
|
|
|
#endif // 0
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Sum up the sizes of all sub-tables. */
|
|
|
|
/***********************************************************************/
|
|
|
|
int TDBTBL::GetMaxSize(PGLOBAL g)
|
|
|
|
{
|
|
|
|
if (MaxSize < 0) {
|
|
|
|
int mxsz;
|
|
|
|
|
|
|
|
if (!Tablist && InitTableList(g))
|
|
|
|
return 0; // Cannot be calculated at this stage
|
|
|
|
|
|
|
|
// if (Use == USE_OPEN) {
|
|
|
|
// strcpy(g->Message, MSG(MAXSIZE_ERROR));
|
|
|
|
// return -1;
|
|
|
|
// } else
|
|
|
|
MaxSize = 0;
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
for (PTABLE tabp = Tablist; tabp; tabp = tabp->GetNext()) {
|
|
|
|
if ((mxsz = tabp->GetTo_Tdb()->GetMaxSize(g)) < 0) {
|
2013-02-07 13:34:27 +04:00
|
|
|
MaxSize = -1;
|
|
|
|
return mxsz;
|
|
|
|
} // endif mxsz
|
|
|
|
|
|
|
|
MaxSize += mxsz;
|
|
|
|
} // endfor i
|
|
|
|
|
|
|
|
} // endif MaxSize
|
|
|
|
|
|
|
|
return MaxSize;
|
|
|
|
} // end of GetMaxSize
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Reset read/write position values. */
|
|
|
|
/***********************************************************************/
|
|
|
|
void TDBTBL::ResetDB(void)
|
|
|
|
{
|
|
|
|
for (PCOL colp = Columns; colp; colp = colp->GetNext())
|
|
|
|
if (colp->GetAmType() == TYPE_AM_TABID)
|
|
|
|
colp->COLBLK::Reset();
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
for (PTABLE tabp = Tablist; tabp; tabp = tabp->GetNext())
|
|
|
|
((PTDBASE)tabp->GetTo_Tdb())->ResetDB();
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
Tdbp = (PTDBASE)Tablist->GetTo_Tdb();
|
|
|
|
Crp = 0;
|
|
|
|
} // end of ResetDB
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Returns RowId if b is false or Rownum if b is true. */
|
|
|
|
/***********************************************************************/
|
|
|
|
int TDBTBL::RowNumber(PGLOBAL g, bool b)
|
|
|
|
{
|
|
|
|
return Tdbp->RowNumber(g) + ((b) ? 0 : Rows);
|
|
|
|
} // end of RowNumber
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* TBL Access Method opening routine. */
|
|
|
|
/* Open first file, other will be opened sequencially when reading. */
|
|
|
|
/***********************************************************************/
|
|
|
|
bool TDBTBL::OpenDB(PGLOBAL g)
|
|
|
|
{
|
|
|
|
if (trace)
|
|
|
|
htrc("TBL OpenDB: tdbp=%p tdb=R%d use=%d key=%p mode=%d\n",
|
|
|
|
this, Tdb_No, Use, To_Key_Col, Mode);
|
|
|
|
|
|
|
|
if (Use == USE_OPEN) {
|
|
|
|
/*******************************************************************/
|
|
|
|
/* Table already open, replace it at its beginning. */
|
|
|
|
/*******************************************************************/
|
|
|
|
ResetDB();
|
|
|
|
return Tdbp->OpenDB(g); // Re-open fist table
|
|
|
|
} // endif use
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/*********************************************************************/
|
|
|
|
/* Direct access needed for join or sorting. */
|
|
|
|
/*********************************************************************/
|
|
|
|
if (NeedIndexing(g)) {
|
|
|
|
// Direct access of TBL tables is not implemented yet
|
|
|
|
strcpy(g->Message, MSG(NO_MUL_DIR_ACC));
|
|
|
|
return TRUE;
|
|
|
|
} // endif NeedIndexing
|
|
|
|
#endif // 0
|
|
|
|
|
|
|
|
/*********************************************************************/
|
|
|
|
/* When GetMaxsize was called, To_Filter was not set yet. */
|
|
|
|
/*********************************************************************/
|
|
|
|
if (To_Filter && Tablist) {
|
|
|
|
Tablist = NULL;
|
|
|
|
Nbf = 0;
|
|
|
|
} // endif To_Filter
|
|
|
|
|
|
|
|
/*********************************************************************/
|
|
|
|
/* Open the first table of the list. */
|
|
|
|
/*********************************************************************/
|
|
|
|
if (!Tablist && InitTableList(g)) // done in GetMaxSize
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if ((CurTable = Tablist)) {
|
|
|
|
Tdbp = (PTDBASE)CurTable->GetTo_Tdb();
|
|
|
|
Tdbp->SetMode(Mode);
|
|
|
|
// Tdbp->ResetDB();
|
|
|
|
// Tdbp->ResetSize();
|
|
|
|
|
|
|
|
// Check and initialize the subtable columns
|
|
|
|
for (PCOL cp = Columns; cp; cp = cp->GetNext())
|
|
|
|
if (cp->GetAmType() == TYPE_AM_TABID)
|
|
|
|
cp->COLBLK::Reset();
|
2013-04-29 13:50:20 +02:00
|
|
|
else if (((PPRXCOL)cp)->Init(g))
|
2013-02-07 13:34:27 +04:00
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (trace)
|
|
|
|
htrc("Opening subtable %s\n", Tdbp->GetName());
|
|
|
|
|
|
|
|
// Now we can safely open the table
|
|
|
|
if (Tdbp->OpenDB(g))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
} // endif *Tablist
|
|
|
|
|
|
|
|
Use = USE_OPEN;
|
|
|
|
return FALSE;
|
|
|
|
} // end of OpenDB
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* ReadDB: Data Base read routine for MUL access method. */
|
|
|
|
/***********************************************************************/
|
|
|
|
int TDBTBL::ReadDB(PGLOBAL g)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (!CurTable)
|
|
|
|
return RC_EF;
|
|
|
|
else if (To_Kindex) {
|
|
|
|
/*******************************************************************/
|
|
|
|
/* Reading is by an index table. */
|
|
|
|
/*******************************************************************/
|
|
|
|
strcpy(g->Message, MSG(NO_INDEX_READ));
|
|
|
|
rc = RC_FX;
|
|
|
|
} else {
|
|
|
|
/*******************************************************************/
|
|
|
|
/* Now start the reading process. */
|
|
|
|
/*******************************************************************/
|
|
|
|
retry:
|
|
|
|
rc = Tdbp->ReadDB(g);
|
|
|
|
|
|
|
|
if (rc == RC_EF) {
|
|
|
|
// Total number of rows met so far
|
|
|
|
Rows += Tdbp->RowNumber(g) - 1;
|
|
|
|
Crp += Tdbp->GetProgMax(g);
|
|
|
|
|
|
|
|
if ((CurTable = CurTable->GetNext())) {
|
|
|
|
/***************************************************************/
|
|
|
|
/* Continue reading from next table file. */
|
|
|
|
/***************************************************************/
|
|
|
|
Tdbp->CloseDB(g);
|
|
|
|
Tdbp = (PTDBASE)CurTable->GetTo_Tdb();
|
|
|
|
|
|
|
|
// Check and initialize the subtable columns
|
|
|
|
for (PCOL cp = Columns; cp; cp = cp->GetNext())
|
|
|
|
if (cp->GetAmType() == TYPE_AM_TABID)
|
|
|
|
cp->COLBLK::Reset();
|
2013-04-29 13:50:20 +02:00
|
|
|
else if (((PPRXCOL)cp)->Init(g))
|
2013-02-07 13:34:27 +04:00
|
|
|
return RC_FX;
|
|
|
|
|
|
|
|
if (trace)
|
|
|
|
htrc("Opening subtable %s\n", Tdbp->GetName());
|
|
|
|
|
|
|
|
// Now we can safely open the table
|
|
|
|
if (Tdbp->OpenDB(g)) // Open next table
|
|
|
|
return RC_FX;
|
|
|
|
|
|
|
|
goto retry;
|
|
|
|
} // endif iFile
|
|
|
|
|
|
|
|
} else if (rc == RC_FX)
|
|
|
|
strcat(strcat(strcat(g->Message, " ("), Tdbp->GetName()), ")");
|
|
|
|
|
|
|
|
} // endif To_Kindex
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
} // end of ReadDB
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
#if 0
|
2013-02-07 13:34:27 +04:00
|
|
|
/***********************************************************************/
|
|
|
|
/* Data Base write routine for MUL access method. */
|
|
|
|
/***********************************************************************/
|
|
|
|
int TDBTBL::WriteDB(PGLOBAL g)
|
|
|
|
{
|
|
|
|
strcpy(g->Message, MSG(TABMUL_READONLY));
|
|
|
|
return RC_FX; // NIY
|
|
|
|
} // end of WriteDB
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Data Base delete line routine for MUL access method. */
|
|
|
|
/***********************************************************************/
|
|
|
|
int TDBTBL::DeleteDB(PGLOBAL g, int irc)
|
|
|
|
{
|
|
|
|
strcpy(g->Message, MSG(TABMUL_READONLY));
|
|
|
|
return RC_FX; // NIY
|
|
|
|
} // end of DeleteDB
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Data Base close routine for MUL access method. */
|
|
|
|
/***********************************************************************/
|
|
|
|
void TDBTBL::CloseDB(PGLOBAL g)
|
|
|
|
{
|
|
|
|
if (Tdbp)
|
|
|
|
Tdbp->CloseDB(g);
|
|
|
|
|
|
|
|
} // end of CloseDB
|
2013-04-29 13:50:20 +02:00
|
|
|
#endif // 0
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
/* ---------------------------- TBLCOL ------------------------------- */
|
|
|
|
|
2013-04-29 13:50:20 +02:00
|
|
|
#if 0
|
2013-02-07 13:34:27 +04:00
|
|
|
/***********************************************************************/
|
|
|
|
/* TBLCOL public constructor. */
|
|
|
|
/***********************************************************************/
|
|
|
|
TBLCOL::TBLCOL(PCOLDEF cdp, PTDB tdbp, PCOL cprec, int i, PSZ am)
|
|
|
|
: COLBLK(cdp, tdbp, i)
|
|
|
|
{
|
|
|
|
if (cprec) {
|
|
|
|
Next = cprec->GetNext();
|
|
|
|
cprec->SetNext(this);
|
|
|
|
} else {
|
|
|
|
Next = tdbp->GetColumns();
|
|
|
|
tdbp->SetColumns(this);
|
|
|
|
} // endif cprec
|
|
|
|
|
|
|
|
// Set additional Dos access method information for column.
|
|
|
|
Long = cdp->GetLong(); // ???
|
|
|
|
//strcpy(F_Date, cdp->F_Date);
|
|
|
|
Colp = NULL;
|
|
|
|
To_Val = NULL;
|
|
|
|
Pseudo = FALSE;
|
|
|
|
Colnum = cdp->GetOffset(); // If columns are retrieved by number
|
|
|
|
|
|
|
|
if (trace)
|
|
|
|
htrc(" making new %sCOL C%d %s at %p\n", am, Index, Name, this);
|
|
|
|
|
|
|
|
} // end of TBLCOL constructor
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* TBLCOL public constructor. */
|
|
|
|
/***********************************************************************/
|
|
|
|
TBLCOL::TBLCOL(SPCBLK *scp, PTDB tdbp) : COLBLK(scp->GetName(), tdbp, 0)
|
|
|
|
{
|
|
|
|
// Set additional TBL access method information for pseudo column.
|
|
|
|
Is_Key = Was_Key = scp->IsKey();
|
|
|
|
Long = scp->GetLength();
|
|
|
|
Buf_Type = scp->GetResultType();
|
|
|
|
*Format.Type = (Buf_Type == TYPE_INT) ? 'N' : 'C';
|
|
|
|
Format.Length = Long;
|
|
|
|
Colp = NULL;
|
|
|
|
To_Val = NULL;
|
|
|
|
Pseudo = TRUE;
|
|
|
|
} // end of TBLCOL constructor
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* TBLCOL constructor used for copying columns. */
|
|
|
|
/* tdbp is the pointer to the new table descriptor. */
|
|
|
|
/***********************************************************************/
|
|
|
|
TBLCOL::TBLCOL(TBLCOL *col1, PTDB tdbp) : COLBLK(col1, tdbp)
|
|
|
|
{
|
|
|
|
Long = col1->Long;
|
|
|
|
Colp = col1->Colp;
|
|
|
|
To_Val = col1->To_Val;
|
|
|
|
Pseudo = col1->Pseudo;
|
|
|
|
} // end of TBLCOL copy constructor
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* TBLCOL initialization routine. */
|
|
|
|
/* Look for the matching column in the current table. */
|
|
|
|
/***********************************************************************/
|
|
|
|
bool TBLCOL::Init(PGLOBAL g)
|
|
|
|
{
|
|
|
|
PTDBTBL tdbp = (PTDBTBL)To_Tdb;
|
|
|
|
|
|
|
|
To_Val = NULL;
|
|
|
|
|
|
|
|
if (!(Colp = tdbp->Tdbp->ColDB(g, Name, 0)) && Colnum)
|
|
|
|
Colp = tdbp->Tdbp->ColDB(g, NULL, Colnum);
|
|
|
|
|
|
|
|
if (Colp) {
|
|
|
|
Colp->InitValue(g); // May not have been done elsewhere
|
|
|
|
To_Val = Colp->GetValue();
|
|
|
|
} else if (!tdbp->Accept) {
|
|
|
|
sprintf(g->Message, MSG(NO_MATCHING_COL), Name, tdbp->Tdbp->GetName());
|
|
|
|
return TRUE;
|
2013-04-12 18:30:15 +02:00
|
|
|
} else {
|
|
|
|
if (Nullable)
|
|
|
|
Value->SetNull(true);
|
|
|
|
|
2013-02-07 13:34:27 +04:00
|
|
|
Value->Reset();
|
2013-04-12 18:30:15 +02:00
|
|
|
} // endif's
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
} // end of Init
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* ReadColumn: */
|
|
|
|
/***********************************************************************/
|
|
|
|
void TBLCOL::ReadColumn(PGLOBAL g)
|
|
|
|
{
|
|
|
|
if (trace)
|
|
|
|
htrc("TBL ReadColumn: name=%s\n", Name);
|
|
|
|
|
|
|
|
if (Colp) {
|
|
|
|
Colp->ReadColumn(g);
|
|
|
|
Value->SetValue_pval(To_Val);
|
2013-04-10 23:38:27 +02:00
|
|
|
|
|
|
|
// Set null when applicable
|
2013-04-12 18:30:15 +02:00
|
|
|
if (Nullable)
|
|
|
|
Value->SetNull(Value->IsNull());
|
2013-04-10 23:38:27 +02:00
|
|
|
|
2013-02-07 13:34:27 +04:00
|
|
|
} // endif Colp
|
|
|
|
|
|
|
|
} // end of ReadColumn
|
2013-04-29 13:50:20 +02:00
|
|
|
#endif // 0
|
2013-02-07 13:34:27 +04:00
|
|
|
|
|
|
|
/* ---------------------------- TBTBLK ------------------------------- */
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* ReadColumn: */
|
|
|
|
/***********************************************************************/
|
|
|
|
void TBTBLK::ReadColumn(PGLOBAL g)
|
|
|
|
{
|
|
|
|
if (trace)
|
|
|
|
htrc("TBT ReadColumn: name=%s\n", Name);
|
|
|
|
|
|
|
|
Value->SetValue_psz((char*)((PTDBTBL)To_Tdb)->Tdbp->GetName());
|
2013-04-10 23:38:27 +02:00
|
|
|
|
2013-02-07 13:34:27 +04:00
|
|
|
} // end of ReadColumn
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------- */
|