Mozilla/mozilla/extensions/sql/odbc/src/mozSqlConnectionODBC.cpp
Jan.Varga%gmail.com 9a363418a9 SQLConnect() doesn't work with some ODBC drivers.
git-svn-id: svn://10.0.0.236/trunk@199099 18797224-902f-48f8-a5cc-f745e15eee43
2006-06-06 10:41:38 +00:00

227 lines
6.7 KiB
C++

/* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (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.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is mozilla.org code.
*
* The Initial Developer of the Original Code is mozilla.org
* Portions created by the Initial Developer are Copyright (C) 2006
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
*
* Alternatively, the contents of this file may be used under the terms of
* either the GNU General Public License Version 2 or later (the "GPL"), or
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
* ***** END LICENSE BLOCK ***** */
#include "nsReadableUtils.h"
#include "mozSqlConnectionODBC.h"
#include "mozSqlResultODBC.h"
mozSqlConnectionODBC::mozSqlConnectionODBC()
: mConnection(nsnull)
{
}
mozSqlConnectionODBC::~mozSqlConnectionODBC()
{
if (mConnection) {
SQLDisconnect(mConnection);
SQLFreeHandle(SQL_HANDLE_DBC, mConnection);
SQLFreeHandle(SQL_HANDLE_ENV, mEnv);
}
}
NS_IMPL_ADDREF_INHERITED(mozSqlConnectionODBC, mozSqlConnection)
NS_IMPL_RELEASE_INHERITED(mozSqlConnectionODBC, mozSqlConnection)
// QueryInterface
NS_INTERFACE_MAP_BEGIN(mozSqlConnectionODBC)
NS_INTERFACE_MAP_ENTRY(mozISqlConnectionODBC)
NS_INTERFACE_MAP_END_INHERITING(mozSqlConnection)
NS_IMETHODIMP
mozSqlConnectionODBC::Init(const nsAString &aHost, PRInt32 aPort,
const nsAString &aDatabase, const nsAString &aUsername,
const nsAString &aPassword)
{
if (mConnection)
return NS_OK;
SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &mEnv);
SQLSetEnvAttr(mEnv, SQL_ATTR_ODBC_VERSION, (void *) SQL_OV_ODBC3, 0);
SQLAllocHandle(SQL_HANDLE_DBC, mEnv, &mConnection);
NS_LossyConvertUTF16toASCII inString(NS_LITERAL_STRING("DSN=") + aDatabase +
NS_LITERAL_STRING(";UID=") + aUsername +
NS_LITERAL_STRING(";PWD=") + aPassword);
char outString[1024];
SQLSMALLINT outStringLength;
if (!SQL_SUCCEEDED(SQLDriverConnect(mConnection, NULL,
(SQLCHAR*)inString.get(),
inString.Length(),
(SQLCHAR*)outString, sizeof(outString),
&outStringLength,
SQL_DRIVER_COMPLETE))) {
SetError(mConnection, SQL_HANDLE_DBC);
return NS_ERROR_FAILURE;
}
return NS_OK;
}
NS_IMETHODIMP
mozSqlConnectionODBC::GetPrimaryKeys(const nsAString& aSchema,
const nsAString& aTable,
mozISqlResult** _retval)
{
//SQLPrimaryKeys isn't implemented in most drivers
SQLHSTMT hstmt;
SQLAllocHandle(SQL_HANDLE_STMT, mConnection, &hstmt);
if(!SQL_SUCCEEDED(SQLSpecialColumns(hstmt, SQL_BEST_ROWID, NULL, 0,
(SQLCHAR*) NULL, SQL_NTS,
(SQLCHAR*) NS_LossyConvertUTF16toASCII(aTable).get(), SQL_NTS, 0, 0))){
SetError(hstmt, SQL_HANDLE_STMT);
return NS_ERROR_FAILURE;
}
nsAutoString select;
PRInt32 i = 0;
select.AssignLiteral("");
while (SQL_SUCCEEDED(SQLFetch(hstmt))) {
i++;
if(i > 1) select.AppendLiteral(" UNION ");
select.AppendLiteral("select NULL as TABLE_SCHEM, '");
select.Append(aTable);
select.AppendLiteral("' as TABLE_NAME, '");
SQLCHAR colname[512];
SQLGetData(hstmt, 2, SQL_C_CHAR, colname, sizeof(colname), NULL);
select.AppendASCII((char*)colname);
select.AppendLiteral("' as COLUMN_NAME, ");
char str[16];
sprintf(str, "%d", i);
select.AppendLiteral(str);
select.AppendLiteral(" as KEY_SEQ, NULL as PK_NAME");
}
if(select.IsEmpty()) {
select.AppendLiteral("SELECT * FROM ");
select.Append(aTable);
select.AppendLiteral(" WHERE 1 = 2");
}
select.AppendLiteral(";");
return RealExec(select, _retval, nsnull);
}
nsresult
mozSqlConnectionODBC::RealExec(const nsAString& aQuery,
mozISqlResult** aResult, PRInt32* aAffectedRows)
{
if (!mConnection)
return NS_ERROR_NOT_INITIALIZED;
SQLHSTMT hstmt;
SQLAllocHandle(SQL_HANDLE_STMT, mConnection, &hstmt);
if(!SQL_SUCCEEDED(SQLExecDirect(hstmt, (SQLCHAR*) NS_LossyConvertUTF16toASCII(aQuery).get(), SQL_NTS))){
SetError(hstmt, SQL_HANDLE_STMT);
return NS_ERROR_FAILURE;
}
if (!aResult){
if (!aAffectedRows)
return NS_ERROR_NULL_POINTER;
SQLINTEGER count;
SQLRowCount(hstmt, &count);
*aAffectedRows = (PRInt32) count;
return NS_OK;
}
mozSqlResult *mozresult = new mozSqlResultODBC(this, aQuery);
if (!mozresult){
SQLFreeHandle(SQL_HANDLE_STMT, hstmt);
return NS_ERROR_OUT_OF_MEMORY;
}
((mozSqlResultODBC*)mozresult)->SetResult(hstmt);
nsresult rv = mozresult->Init();
if (NS_FAILED(rv)){
delete mozresult;
return rv;
}
*aResult = mozresult;
NS_ADDREF(*aResult);
return NS_OK;
}
nsresult
mozSqlConnectionODBC::CancelExec()
{
// SQLCancel needs the SQLHSTMT, other ways?
return NS_ERROR_NOT_IMPLEMENTED;
}
nsresult
mozSqlConnectionODBC::GetIDName(nsAString& aIDName)
{
// not sure what to return here
aIDName.AssignLiteral("");
return NS_OK;
}
void
mozSqlConnectionODBC::SetError(SQLHANDLE aHandle, SQLSMALLINT aType)
{
SQLCHAR state[6], text[SQL_MAX_MESSAGE_LENGTH];
SQLINTEGER native;
char err[256];
SQLINTEGER i = 1;
mErrorMessage.Assign(NS_LITERAL_STRING(""));
while (SQL_SUCCEEDED(SQLGetDiagRec(aType, aHandle, i++, state, &native, text, sizeof(text), NULL))) {
sprintf((char *)err,"%s\nState: %s; Native Error: %lX\n", text, state, native);
mErrorMessage.Append(ToNewUnicode(nsDependentCString((char *)err)));
}
}