// Copyright (C) 1999-2005 Open Source Telecom Corporation.
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 2 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
//
// As a special exception, you may use this file as part of a free software
// library without restriction. Specifically, if other files instantiate
// templates or use macros or inline functions from this file, or you compile
// this file and link it with other files to produce an executable, this
// file does not by itself cause the resulting executable to be covered by
// the GNU General Public License. This exception does not however
// invalidate any other reasons why the executable file might be covered by
// the GNU General Public License.
//
// This exception applies only to the code released under the name GNU
// ccScript. If you copy code from other releases into a copy of GNU
// ccScript, as the General Public License permits, the exception does
// not apply to the code that you add in this way. To avoid misleading
// anyone as to the status of such modified files, you must delete
// this exception notice from them.
//
// If you write modifications of your own for GNU ccScript, it is your choice
// whether to permit this exception to apply to your modifications.
// If you do not wish that, delete this exception notice.
//
#include "module.h"
using namespace ccscript3Database;
using namespace std;
using namespace ost;
SQLThread::SQLThread(ScriptInterp *interp, SQLDatabase *c, Symbol *r) :
ScriptThread(interp, 0, 64000l + 32000l * sizeof(void *))
{
current = c;
sym = r;
hstmt = NULL;
}
SQLThread::~SQLThread()
{
terminate();
if(hstmt)
SQLFreeHandle(SQL_HANDLE_STMT, hstmt);
}
void SQLThread::run(void)
{
const char *opt;
SQLINTEGER err;
SQLCHAR stat[10];
SQLCHAR errmsg[128];
SQLSMALLINT mlen, col, cols;
#if ODBCVER >= 0x0300 && !defined(__ppc__)
SQLINTEGER rowcnt;
#else
long int rowcnt;
#endif
char buf[1024];
char nbuf[12];
unsigned row = 0, len = 0;
if(!current->isConnected())
{
lock();
interp->setSymbol("sql.error", "sql-connect-failed");
release();
exit("sql-connect-failed");
}
lock();
interp->setSymbol("sql.database", current->getName());
interp->setSymbol("sql.error", "none");
interp->setSymbol("sql.rows", "0");
release();
err = SQLAllocHandle(SQL_HANDLE_STMT, current->hdbc, &hstmt);
if(err != SQL_SUCCESS && err != SQL_SUCCESS_WITH_INFO)
{
hstmt = NULL;
SQLGetDiagRec(SQL_HANDLE_DBC, current->hdbc, 1,
stat, &err, errmsg, sizeof(errmsg), &mlen);
lock();
interp->setSymbol("sql.error", (char *)errmsg + 4);
release();
slog.error("odbc: %s: %s", current->name, errmsg);
exit("sql-prep-failed");
}
buf[0] = 0;
while(NULL != (opt = interp->getValue(NULL)))
{
Thread::yield();
addString(buf, sizeof(buf), opt);
}
Thread::yield();
err = SQLExecDirect(hstmt, (SQLCHAR *)buf, SQL_NTS);
if(err != SQL_SUCCESS && err != SQL_SUCCESS_WITH_INFO)
{
strcpy((char *)errmsg, "sql.");
SQLGetDiagRec(SQL_HANDLE_DBC, current->hdbc, 1,
stat, &err, errmsg, sizeof(errmsg) / sizeof(SQLCHAR), &mlen);
lock();
interp->setSymbol("sql.error", (char *)errmsg);
release();
slog.error("odbc: %s: %s", current->name, errmsg);
exit("sql-query-failed");
}
Thread::yield();
SQLRowCount(hstmt, &rowcnt);
snprintf(buf, sizeof(buf), "%d", (int)rowcnt);
lock();
interp->setNumber("sql.rows", buf);
release();
if(!sym)
exit(NULL);
while(row < count(sym))
{
err = SQLFetch(hstmt);
if(err != SQL_SUCCESS && err != SQL_SUCCESS_WITH_INFO)
break;
Thread::yield();
snprintf(nbuf, sizeof(nbuf), "%d", ++row);
SQLNumResultCols(hstmt, &cols);
if(cols < 1)
break;
col = 0;
len = 0;
while(col < cols && len < sizeof(buf) - 3)
{
if(len)
buf[len++] = ',';
buf[len++] = '\'';
SQLGetData(hstmt, ++col, SQL_C_CHAR, buf + len, (SQLINTEGER)(256 - len), &rowcnt);
len = (unsigned)strlen(buf);
buf[len++] = '\'';
Thread::yield();
}
commit(sym, buf);
Thread::yield();
}
// post actual row count
snprintf(nbuf, sizeof(nbuf), "%d", row);
lock();
interp->setNumber("sql.rows", nbuf);
release();
exit(NULL);
}
syntax highlighted by Code2HTML, v. 0.9.1