// 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